2019-05-03 18:45:21 +00:00
/* SleepLib PRS1 Loader Implementation
2014-04-09 21:01:57 +00:00
*
2020-01-07 02:45:52 +00:00
* Copyright ( c ) 2019 - 2020 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
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>
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"
2011-06-26 08:30:44 +00:00
# include "prs1_loader.h"
# include "SleepLib/session.h"
2011-12-10 12:14:48 +00:00
# include "SleepLib/calcs.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
2019-05-18 23:17:55 +00:00
const int PRS1_HTYPE_NORMAL = 0 ;
const int PRS1_HTYPE_INTERVAL = 1 ;
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.
//********************************************************************************************
2011-11-26 04:00:31 +00:00
2019-05-15 19:16:14 +00:00
// CRC-16/KERMIT, polynomial: 0x11021, bit reverse algorithm
// Table generated by crcmod (crc-kermit)
2011-11-26 04:00:31 +00:00
2019-05-15 19:16:14 +00:00
typedef quint16 crc16_t ;
static crc16_t CRC16 ( unsigned char * data , size_t data_len , crc16_t crc = 0 )
{
static const crc16_t table [ 256 ] = {
0x0000U , 0x1189U , 0x2312U , 0x329bU , 0x4624U , 0x57adU , 0x6536U , 0x74bfU ,
0x8c48U , 0x9dc1U , 0xaf5aU , 0xbed3U , 0xca6cU , 0xdbe5U , 0xe97eU , 0xf8f7U ,
0x1081U , 0x0108U , 0x3393U , 0x221aU , 0x56a5U , 0x472cU , 0x75b7U , 0x643eU ,
0x9cc9U , 0x8d40U , 0xbfdbU , 0xae52U , 0xdaedU , 0xcb64U , 0xf9ffU , 0xe876U ,
0x2102U , 0x308bU , 0x0210U , 0x1399U , 0x6726U , 0x76afU , 0x4434U , 0x55bdU ,
0xad4aU , 0xbcc3U , 0x8e58U , 0x9fd1U , 0xeb6eU , 0xfae7U , 0xc87cU , 0xd9f5U ,
0x3183U , 0x200aU , 0x1291U , 0x0318U , 0x77a7U , 0x662eU , 0x54b5U , 0x453cU ,
0xbdcbU , 0xac42U , 0x9ed9U , 0x8f50U , 0xfbefU , 0xea66U , 0xd8fdU , 0xc974U ,
0x4204U , 0x538dU , 0x6116U , 0x709fU , 0x0420U , 0x15a9U , 0x2732U , 0x36bbU ,
0xce4cU , 0xdfc5U , 0xed5eU , 0xfcd7U , 0x8868U , 0x99e1U , 0xab7aU , 0xbaf3U ,
0x5285U , 0x430cU , 0x7197U , 0x601eU , 0x14a1U , 0x0528U , 0x37b3U , 0x263aU ,
0xdecdU , 0xcf44U , 0xfddfU , 0xec56U , 0x98e9U , 0x8960U , 0xbbfbU , 0xaa72U ,
0x6306U , 0x728fU , 0x4014U , 0x519dU , 0x2522U , 0x34abU , 0x0630U , 0x17b9U ,
0xef4eU , 0xfec7U , 0xcc5cU , 0xddd5U , 0xa96aU , 0xb8e3U , 0x8a78U , 0x9bf1U ,
0x7387U , 0x620eU , 0x5095U , 0x411cU , 0x35a3U , 0x242aU , 0x16b1U , 0x0738U ,
0xffcfU , 0xee46U , 0xdcddU , 0xcd54U , 0xb9ebU , 0xa862U , 0x9af9U , 0x8b70U ,
0x8408U , 0x9581U , 0xa71aU , 0xb693U , 0xc22cU , 0xd3a5U , 0xe13eU , 0xf0b7U ,
0x0840U , 0x19c9U , 0x2b52U , 0x3adbU , 0x4e64U , 0x5fedU , 0x6d76U , 0x7cffU ,
0x9489U , 0x8500U , 0xb79bU , 0xa612U , 0xd2adU , 0xc324U , 0xf1bfU , 0xe036U ,
0x18c1U , 0x0948U , 0x3bd3U , 0x2a5aU , 0x5ee5U , 0x4f6cU , 0x7df7U , 0x6c7eU ,
0xa50aU , 0xb483U , 0x8618U , 0x9791U , 0xe32eU , 0xf2a7U , 0xc03cU , 0xd1b5U ,
0x2942U , 0x38cbU , 0x0a50U , 0x1bd9U , 0x6f66U , 0x7eefU , 0x4c74U , 0x5dfdU ,
0xb58bU , 0xa402U , 0x9699U , 0x8710U , 0xf3afU , 0xe226U , 0xd0bdU , 0xc134U ,
0x39c3U , 0x284aU , 0x1ad1U , 0x0b58U , 0x7fe7U , 0x6e6eU , 0x5cf5U , 0x4d7cU ,
0xc60cU , 0xd785U , 0xe51eU , 0xf497U , 0x8028U , 0x91a1U , 0xa33aU , 0xb2b3U ,
0x4a44U , 0x5bcdU , 0x6956U , 0x78dfU , 0x0c60U , 0x1de9U , 0x2f72U , 0x3efbU ,
0xd68dU , 0xc704U , 0xf59fU , 0xe416U , 0x90a9U , 0x8120U , 0xb3bbU , 0xa232U ,
0x5ac5U , 0x4b4cU , 0x79d7U , 0x685eU , 0x1ce1U , 0x0d68U , 0x3ff3U , 0x2e7aU ,
0xe70eU , 0xf687U , 0xc41cU , 0xd595U , 0xa12aU , 0xb0a3U , 0x8238U , 0x93b1U ,
0x6b46U , 0x7acfU , 0x4854U , 0x59ddU , 0x2d62U , 0x3cebU , 0x0e70U , 0x1ff9U ,
0xf78fU , 0xe606U , 0xd49dU , 0xc514U , 0xb1abU , 0xa022U , 0x92b9U , 0x8330U ,
0x7bc7U , 0x6a4eU , 0x58d5U , 0x495cU , 0x3de3U , 0x2c6aU , 0x1ef1U , 0x0f78U ,
} ;
2011-11-26 04:00:31 +00:00
2019-05-15 19:16:14 +00:00
for ( size_t i = 0 ; i < data_len ; i + + ) {
crc = table [ ( * data ^ ( unsigned char ) crc ) & 0xFF ] ^ ( crc > > 8 ) ;
2011-11-26 04:00:31 +00:00
data + + ;
}
2019-05-15 19:16:14 +00:00
return crc ;
2011-11-26 04:00:31 +00:00
}
2019-05-15 19:16:14 +00:00
2011-11-26 04:00:31 +00:00
2019-05-15 21:41:37 +00:00
// CRC-32/MPEG-2, polynomial: 0x104C11DB7
// Table generated by crcmod (crc-32-mpeg)
typedef quint32 crc32_t ;
static crc32_t CRC32 ( const unsigned char * data , size_t data_len , crc32_t crc = 0xffffffffU )
{
static const crc32_t table [ 256 ] = {
0x00000000U , 0x04c11db7U , 0x09823b6eU , 0x0d4326d9U ,
0x130476dcU , 0x17c56b6bU , 0x1a864db2U , 0x1e475005U ,
0x2608edb8U , 0x22c9f00fU , 0x2f8ad6d6U , 0x2b4bcb61U ,
0x350c9b64U , 0x31cd86d3U , 0x3c8ea00aU , 0x384fbdbdU ,
0x4c11db70U , 0x48d0c6c7U , 0x4593e01eU , 0x4152fda9U ,
0x5f15adacU , 0x5bd4b01bU , 0x569796c2U , 0x52568b75U ,
0x6a1936c8U , 0x6ed82b7fU , 0x639b0da6U , 0x675a1011U ,
0x791d4014U , 0x7ddc5da3U , 0x709f7b7aU , 0x745e66cdU ,
0x9823b6e0U , 0x9ce2ab57U , 0x91a18d8eU , 0x95609039U ,
0x8b27c03cU , 0x8fe6dd8bU , 0x82a5fb52U , 0x8664e6e5U ,
0xbe2b5b58U , 0xbaea46efU , 0xb7a96036U , 0xb3687d81U ,
0xad2f2d84U , 0xa9ee3033U , 0xa4ad16eaU , 0xa06c0b5dU ,
0xd4326d90U , 0xd0f37027U , 0xddb056feU , 0xd9714b49U ,
0xc7361b4cU , 0xc3f706fbU , 0xceb42022U , 0xca753d95U ,
0xf23a8028U , 0xf6fb9d9fU , 0xfbb8bb46U , 0xff79a6f1U ,
0xe13ef6f4U , 0xe5ffeb43U , 0xe8bccd9aU , 0xec7dd02dU ,
0x34867077U , 0x30476dc0U , 0x3d044b19U , 0x39c556aeU ,
0x278206abU , 0x23431b1cU , 0x2e003dc5U , 0x2ac12072U ,
0x128e9dcfU , 0x164f8078U , 0x1b0ca6a1U , 0x1fcdbb16U ,
0x018aeb13U , 0x054bf6a4U , 0x0808d07dU , 0x0cc9cdcaU ,
0x7897ab07U , 0x7c56b6b0U , 0x71159069U , 0x75d48ddeU ,
0x6b93dddbU , 0x6f52c06cU , 0x6211e6b5U , 0x66d0fb02U ,
0x5e9f46bfU , 0x5a5e5b08U , 0x571d7dd1U , 0x53dc6066U ,
0x4d9b3063U , 0x495a2dd4U , 0x44190b0dU , 0x40d816baU ,
0xaca5c697U , 0xa864db20U , 0xa527fdf9U , 0xa1e6e04eU ,
0xbfa1b04bU , 0xbb60adfcU , 0xb6238b25U , 0xb2e29692U ,
0x8aad2b2fU , 0x8e6c3698U , 0x832f1041U , 0x87ee0df6U ,
0x99a95df3U , 0x9d684044U , 0x902b669dU , 0x94ea7b2aU ,
0xe0b41de7U , 0xe4750050U , 0xe9362689U , 0xedf73b3eU ,
0xf3b06b3bU , 0xf771768cU , 0xfa325055U , 0xfef34de2U ,
0xc6bcf05fU , 0xc27dede8U , 0xcf3ecb31U , 0xcbffd686U ,
0xd5b88683U , 0xd1799b34U , 0xdc3abdedU , 0xd8fba05aU ,
0x690ce0eeU , 0x6dcdfd59U , 0x608edb80U , 0x644fc637U ,
0x7a089632U , 0x7ec98b85U , 0x738aad5cU , 0x774bb0ebU ,
0x4f040d56U , 0x4bc510e1U , 0x46863638U , 0x42472b8fU ,
0x5c007b8aU , 0x58c1663dU , 0x558240e4U , 0x51435d53U ,
0x251d3b9eU , 0x21dc2629U , 0x2c9f00f0U , 0x285e1d47U ,
0x36194d42U , 0x32d850f5U , 0x3f9b762cU , 0x3b5a6b9bU ,
0x0315d626U , 0x07d4cb91U , 0x0a97ed48U , 0x0e56f0ffU ,
0x1011a0faU , 0x14d0bd4dU , 0x19939b94U , 0x1d528623U ,
0xf12f560eU , 0xf5ee4bb9U , 0xf8ad6d60U , 0xfc6c70d7U ,
0xe22b20d2U , 0xe6ea3d65U , 0xeba91bbcU , 0xef68060bU ,
0xd727bbb6U , 0xd3e6a601U , 0xdea580d8U , 0xda649d6fU ,
0xc423cd6aU , 0xc0e2d0ddU , 0xcda1f604U , 0xc960ebb3U ,
0xbd3e8d7eU , 0xb9ff90c9U , 0xb4bcb610U , 0xb07daba7U ,
0xae3afba2U , 0xaafbe615U , 0xa7b8c0ccU , 0xa379dd7bU ,
0x9b3660c6U , 0x9ff77d71U , 0x92b45ba8U , 0x9675461fU ,
0x8832161aU , 0x8cf30badU , 0x81b02d74U , 0x857130c3U ,
0x5d8a9099U , 0x594b8d2eU , 0x5408abf7U , 0x50c9b640U ,
0x4e8ee645U , 0x4a4ffbf2U , 0x470cdd2bU , 0x43cdc09cU ,
0x7b827d21U , 0x7f436096U , 0x7200464fU , 0x76c15bf8U ,
0x68860bfdU , 0x6c47164aU , 0x61043093U , 0x65c52d24U ,
0x119b4be9U , 0x155a565eU , 0x18197087U , 0x1cd86d30U ,
0x029f3d35U , 0x065e2082U , 0x0b1d065bU , 0x0fdc1becU ,
0x3793a651U , 0x3352bbe6U , 0x3e119d3fU , 0x3ad08088U ,
0x2497d08dU , 0x2056cd3aU , 0x2d15ebe3U , 0x29d4f654U ,
0xc5a92679U , 0xc1683bceU , 0xcc2b1d17U , 0xc8ea00a0U ,
0xd6ad50a5U , 0xd26c4d12U , 0xdf2f6bcbU , 0xdbee767cU ,
0xe3a1cbc1U , 0xe760d676U , 0xea23f0afU , 0xeee2ed18U ,
0xf0a5bd1dU , 0xf464a0aaU , 0xf9278673U , 0xfde69bc4U ,
0x89b8fd09U , 0x8d79e0beU , 0x803ac667U , 0x84fbdbd0U ,
0x9abc8bd5U , 0x9e7d9662U , 0x933eb0bbU , 0x97ffad0cU ,
0xafb010b1U , 0xab710d06U , 0xa6322bdfU , 0xa2f33668U ,
0xbcb4666dU , 0xb8757bdaU , 0xb5365d03U , 0xb1f740b4U ,
} ;
for ( size_t i = 0 ; i < data_len ; i + + ) {
crc = table [ ( * data ^ ( unsigned char ) ( crc > > 24 ) ) & 0xFF ] ^ ( crc < < 8 ) ;
data + + ;
}
return crc ;
}
// Strangely, the PRS1 CRC32 appears to consider every byte a 32-bit wchar_t.
// Nothing like trying a bunch of encodings and CRC32 variants on PROP.TXT files
// until you find a winner.
static crc32_t CRC32wchar ( const unsigned char * data , size_t data_len , crc32_t crc = 0xffffffffU )
{
for ( size_t i = 0 ; i < data_len ; i + + ) {
static unsigned char wch [ 4 ] = { 0 , 0 , 0 , 0 } ;
wch [ 3 ] = * data + + ;
crc = CRC32 ( wch , 4 , crc ) ;
}
return crc ;
}
2019-09-30 14:23:28 +00:00
static QString ts ( qint64 msecs )
{
// TODO: make this UTC so that tests don't vary by where they're run
return QDateTime : : fromMSecsSinceEpoch ( msecs ) . toString ( Qt : : ISODate ) ;
}
2020-01-05 01:47:28 +00:00
// TODO: See the LogUnexpectedMessage TODO about generalizing this for other loaders.
// Right now this macro assumes that it's called within a method that has a "loader" member
// that points to the PRS1Loader* instance that's calling it.
# define UNEXPECTED_VALUE(SRC, VALS) { \
QString message = QString ( " %1:%2: %3 = %4 != %5 " ) . arg ( __func__ ) . arg ( __LINE__ ) . arg ( # SRC ) . arg ( SRC ) . arg ( VALS ) ; \
qWarning ( ) < < this - > sessionid < < message ; \
loader - > LogUnexpectedMessage ( message ) ; \
}
2019-06-19 20:23:28 +00:00
# define CHECK_VALUE(SRC, VAL) if ((SRC) != (VAL)) UNEXPECTED_VALUE(SRC, VAL)
# define CHECK_VALUES(SRC, VAL1, VAL2) if ((SRC) != (VAL1) && (SRC) != (VAL2)) UNEXPECTED_VALUE(SRC, #VAL1 " or " #VAL2)
// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails
2020-01-10 15:49:08 +00:00
# define HEX(SRC) { qWarning() << this->sessionid << QString("%1:%2: %3 = %4").arg(__func__).arg(__LINE__).arg(#SRC).arg((SRC & 0xFF), 2, 16, QChar('0')); }
2019-06-19 20:23:28 +00:00
2020-01-05 01:47:28 +00:00
void PRS1Loader : : LogUnexpectedMessage ( const QString & message )
{
m_importMutex . lock ( ) ;
m_unexpectedMessages + = message ;
m_importMutex . unlock ( ) ;
}
2019-06-19 20:23:28 +00:00
2019-12-04 00:04:09 +00:00
enum FlexMode { FLEX_None , FLEX_CFlex , FLEX_CFlexPlus , FLEX_AFlex , FLEX_RiseTime , FLEX_BiFlex , FLEX_AVAPS , FLEX_PFlex , FLEX_Unknown } ;
2014-08-03 13:00:13 +00:00
2019-09-23 17:43:28 +00:00
enum BackupBreathMode { PRS1Backup_Off , PRS1Backup_Auto , PRS1Backup_Fixed } ;
2020-01-12 00:14:01 +00:00
enum HumidMode { HUMID_Fixed , HUMID_Adaptive , HUMID_HeatedTube } ;
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 ;
2014-08-03 13:00:13 +00:00
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)
{ " 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) " } ,
{ " 550P " , 0 , 2 , " REMstar Auto (System One) " } ,
{ " 550P " , 0 , 3 , " REMstar Auto (System One) " } ,
{ " 551P " , 0 , 2 , " REMstar Auto (System One) " } ,
{ " 750P " , 0 , 2 , " BiPAP Auto (System One) " } ,
{ " 460P " , 0 , 4 , " REMstar Pro (System One 60 Series) " } ,
{ " 461P " , 0 , 4 , " REMstar Pro (System One 60 Series) " } ,
{ " 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) " } ,
2019-05-27 14:05:16 +00:00
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 " } ,
{ " 400X150 " , 0 , 6 , " DreamStation CPAP Pro " } ,
{ " 500X110 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2020-01-28 18:04:36 +00:00
{ " 500X130 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2019-06-20 04:09:28 +00:00
{ " 500X150 " , 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 " } ,
2019-06-20 04:09:28 +00:00
{ " 502G150 " , 0 , 6 , " DreamStation Go Auto " } ,
{ " 600X110 " , 0 , 6 , " DreamStation BiPAP Pro " } ,
{ " 700X110 " , 0 , 6 , " DreamStation Auto BiPAP " } ,
2019-05-27 14:05:16 +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
{ " 900X110 " , 5 , 3 , " DreamStation BiPAP autoSV " } ,
{ " 900X120 " , 5 , 3 , " DreamStation BiPAP autoSV " } ,
2019-05-27 14:05:16 +00:00
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 " } ,
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
m_bricks = { " 251P " , " 200X110 " } ;
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 ;
} ;
bool PRS1ModelInfo : : IsSupported ( const QHash < QString , QString > & props ) const
{
bool ok ;
int family = props [ " Family " ] . toInt ( & ok , 10 ) ;
if ( ok ) {
int familyVersion = props [ " FamilyVersion " ] . toInt ( & ok , 10 ) ;
if ( ok ) {
ok = IsSupported ( family , familyVersion ) ;
}
}
return ok ;
}
bool PRS1ModelInfo : : IsTested ( const QHash < QString , QString > & props ) const
{
bool ok ;
int family = props [ " Family " ] . toInt ( & ok , 10 ) ;
if ( ok ) {
int familyVersion = props [ " FamilyVersion " ] . toInt ( & ok , 10 ) ;
if ( ok ) {
ok = IsTested ( props [ " ModelNumber " ] , family , familyVersion ) ;
}
}
return ok ;
} ;
2019-06-20 02:19:16 +00:00
bool PRS1ModelInfo : : IsBrick ( const QString & model ) const
{
bool is_brick ;
if ( m_modelNames . contains ( model ) ) {
is_brick = m_bricks . contains ( model ) ;
} else {
// 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
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 " } ,
{ " 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 ( ) ;
// Look for machine directories (containing a PROP.TXT or properties.txt)
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 ( ) ) ) {
// Found a properties file, this is a machine folder
propertyfiles . append ( mfi ) ;
}
}
}
}
// Sort machines from oldest to newest.
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?
{ " 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 ;
}
2014-09-29 14:41:31 +00:00
QTextStream in ( & f ) ;
do {
QString line = in . readLine ( ) ;
QStringList pair = line . split ( " = " ) ;
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 " ;
}
}
props [ pair [ 0 ] ] = pair [ 1 ] ;
} while ( ! in . atEnd ( ) ) ;
return true ;
}
bool PRS1Loader : : PeekProperties ( MachineInfo & info , const QString & filename , Machine * mach )
{
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 ;
}
2014-09-30 05:25:11 +00:00
if ( ! mach | | skip ) continue ;
2019-05-27 14:05:16 +00:00
mach - > properties [ key ] = props [ key ] ;
} ;
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
2020-03-09 15:17:59 +00:00
// Present information about the newest machine on the card.
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 " ) ) {
qWarning ( ) < < " No properties file found in " < < newpath ;
}
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 ) ;
// Return an error if no machines were found.
if ( machines . isEmpty ( ) ) {
qDebug ( ) < < " No PRS1 machines 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
2020-03-09 16:47:54 +00:00
// Import each machine, from oldest to newest.
2014-07-29 14:38:59 +00:00
int c = 0 ;
2020-03-09 16:47:54 +00:00
for ( auto & machinePath : machines ) {
c + = OpenMachine ( machinePath ) ;
2011-06-26 08:30:44 +00:00
}
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
{
2018-04-25 10:34:23 +00:00
if ( p_profile = = nullptr ) {
qWarning ( ) < < " PRS1Loader::OpenMachine() called without a valid p_profile object present " ;
return 0 ;
}
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 ) ;
Machine * m = CreateMachineFromProperties ( propertyfile ) ;
if ( m = = nullptr ) {
return - 1 ;
}
2020-03-09 16:47:54 +00:00
emit updateMessage ( QObject : : tr ( " Backing Up Files... " ) ) ;
QCoreApplication : : processEvents ( ) ;
2019-05-03 20:59:26 +00:00
QString backupPath = m - > getBackupPath ( ) + path . section ( " / " , - 2 ) ;
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.
ScanFiles ( paths , sessionid_base , m ) ;
int tasks = countTasks ( ) ;
emit updateMessage ( QObject : : tr ( " Importing Sessions... " ) ) ;
QCoreApplication : : processEvents ( ) ;
runTasks ( AppSetting - > multithreading ( ) ) ;
emit updateMessage ( QObject : : tr ( " Finishing up... " ) ) ;
QCoreApplication : : processEvents ( ) ;
finishAddingSessions ( ) ;
2020-01-05 03:06:28 +00:00
if ( m_unexpectedMessages . count ( ) > 0 & & p_profile - > session - > warnOnUnexpectedData ( ) ) {
// Compare this to the list of messages previously seen for this machine
// and only alert if there are new ones.
QSet < QString > newMessages = m_unexpectedMessages - m - > previouslySeenUnexpectedData ( ) ;
if ( newMessages . count ( ) > 0 ) {
// TODO: Rework the importer call structure so that this can become an
// emit statement to the appropriate import job.
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
QObject : : tr ( " Untested Data " ) ,
QObject : : tr ( " Your Philips Respironics %1 (%2) generated data that OSCAR has never seen before. " ) . arg ( m - > getInfo ( ) . model ) . arg ( m - > getInfo ( ) . modelnumber ) + " \n \n " +
QObject : : tr ( " The imported data may not be entirely accurate, so the developers would like a .zip copy of this machine's SD card and matching Encore .pdf reports to make sure OSCAR is handling the data correctly. " )
, QMessageBox : : Ok ) ;
m - > previouslySeenUnexpectedData ( ) + = newMessages ;
}
}
2020-01-05 01:47:28 +00:00
2019-05-03 20:59:26 +00:00
return m - > unsupported ( ) ? - 1 : tasks ;
}
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 ( ) ;
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
Machine * PRS1Loader : : CreateMachineFromProperties ( QString propertyfile )
{
2019-05-30 20:32:57 +00:00
QHash < QString , QString > props ;
PeekProperties ( propertyfile , props ) ;
2014-09-30 05:25:11 +00:00
MachineInfo info = newInfo ( ) ;
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
2019-06-20 02:19:16 +00:00
if ( true ) {
if ( s_PRS1ModelInfo . IsBrick ( info . modelnumber ) & & p_profile - > cpap - > brickWarning ( ) ) {
2019-05-05 01:49:50 +00:00
# ifndef UNITTEST_MODE
2014-09-01 11:39:38 +00:00
QApplication : : processEvents ( ) ;
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
2014-05-02 04:34:34 +00:00
QObject : : tr ( " Non Data Capable Machine " ) ,
QString ( QObject : : tr ( " Your Philips Respironics CPAP machine (Model %1) is unfortunately not a data capable model. " ) + " \n \n " +
2019-02-24 00:58:25 +00:00
QObject : : tr ( " I'm sorry to report that OSCAR can only track hours of use and very basic settings for this machine. " ) ) .
2014-09-30 05:25:11 +00:00
arg ( info . modelnumber ) , QMessageBox : : Ok ) ;
2019-05-05 01:49:50 +00:00
# endif
2014-09-01 11:39:38 +00:00
p_profile - > cpap - > setBrickWarning ( false ) ;
2014-05-02 04:34:34 +00:00
}
2014-09-30 05:25:11 +00:00
2019-06-20 02:19:16 +00:00
if ( ! s_PRS1ModelInfo . IsSupported ( props ) ) {
qWarning ( ) < < info . modelnumber < < " unsupported " ;
2019-05-05 01:49:50 +00:00
# ifndef UNITTEST_MODE
2014-09-30 05:25:11 +00:00
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
QObject : : tr ( " Machine Unsupported " ) ,
QObject : : tr ( " Sorry, your Philips Respironics CPAP machine (Model %1) is not supported yet. " ) . arg ( info . modelnumber ) + " \n \n " +
2019-05-30 20:32:57 +00:00
QObject : : tr ( " The developers needs a .zip copy of this machine's SD card and matching Encore .pdf reports to make it work with OSCAR. " )
2014-09-30 05:25:11 +00:00
, QMessageBox : : Ok ) ;
2019-05-05 01:49:50 +00:00
# endif
2019-05-03 20:59:26 +00:00
return nullptr ;
2014-09-30 05:25:11 +00:00
}
}
// Which is needed to get the right machine record..
2018-04-22 12:06:48 +00:00
Machine * m = p_profile - > CreateMachine ( info ) ;
2014-09-30 05:25:11 +00:00
// This time supply the machine object so it can populate machine properties..
PeekProperties ( m - > info , propertyfile , m ) ;
2019-05-30 20:32:57 +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 " ;
2020-01-05 03:06:28 +00:00
if ( p_profile - > session - > warnOnUntestedMachine ( ) & & m - > warnOnUntested ( ) ) {
m - > suppressWarnOnUntested ( ) ; // don't warn the user more than once
2019-05-30 20:32:57 +00:00
# ifndef UNITTEST_MODE
2020-01-05 03:06:28 +00:00
// TODO: Rework the importer call structure so that this can become an
// emit statement to the appropriate import job.
2019-12-30 04:09:46 +00:00
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
2019-05-30 20:32:57 +00:00
QObject : : tr ( " Machine Untested " ) ,
QObject : : tr ( " Your Philips Respironics CPAP machine (Model %1) has not been tested yet. " ) . arg ( info . modelnumber ) + " \n \n " +
QObject : : tr ( " It seems similar enough to other machines that it might work, but the developers would like a .zip copy of this machine's SD card and matching Encore .pdf reports to make sure it works with OSCAR. " )
, QMessageBox : : Ok ) ;
# endif
2019-12-30 04:09:46 +00:00
}
2019-05-30 20:32:57 +00:00
}
2019-05-27 14:05:16 +00:00
2019-06-20 02:19:16 +00:00
// Mark the machine in the profile as unsupported.
2019-05-27 14:05:16 +00:00
if ( ! s_PRS1ModelInfo . IsSupported ( props ) ) {
if ( ! m - > unsupported ( ) ) {
unsupported ( m ) ;
}
}
2019-05-03 20:59:26 +00:00
return m ;
2019-05-03 19:07:15 +00:00
}
2019-10-10 14:34:36 +00:00
static QString relativePath ( const QString & inpath )
{
QStringList pathlist = QDir : : toNativeSeparators ( inpath ) . split ( QDir : : separator ( ) , QString : : SkipEmptyParts ) ;
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 ) ;
}
2019-05-03 19:07:15 +00:00
void PRS1Loader : : ScanFiles ( const QStringList & paths , int sessionid_base , Machine * m )
{
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
2020-01-05 01:47:28 +00:00
m_unexpectedMessages . clear ( ) ;
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 ;
2020-01-21 18:49:02 +00:00
qint64 ignoreBefore = p_profile - > session - > ignoreOlderSessionsDate ( ) . toMSecsSinceEpoch ( ) / 1000 ;
2018-05-07 01:30:42 +00:00
bool ignoreOldSessions = p_profile - > session - > ignoreOlderSessions ( ) ;
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 ) ;
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.
2014-04-17 05:58:57 +00:00
if ( m - > SessionExists ( sid ) ) {
2014-05-02 04:34:34 +00:00
// Skip already imported session
2019-05-13 16:11:04 +00:00
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..
2019-12-01 21:42:54 +00:00
task = new PRS1Import ( this , sid , m , 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.
//
// This seems to happen when the machine begins writing the waveform file
// 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 ) {
2019-10-25 21:27:35 +00:00
if ( ! task - > oxifile . isEmpty ( ) ) {
qDebug ( ) < < sid < < " already has oximetry file " < < relativePath ( task - > oxifile )
< < " skipping " < < relativePath ( fi . canonicalFilePath ( ) ) ;
continue ;
}
2014-09-29 14:41:31 +00:00
task - > oxifile = fi . canonicalFilePath ( ) ;
}
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
2020-01-28 18:04:36 +00:00
// This appears to be benign, probably when a card is out of the machine one night and
// 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
}
2019-06-01 01:47:28 +00:00
if ( m - > 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 {
2019-12-01 21:42:54 +00:00
task = new PRS1Import ( this , chunk_sid , m , 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 ) {
2019-10-10 14:34:36 +00:00
if ( chunksIdentical ( chunk , task - > summary ) ) {
// Never seen identical compliance chunks, so keep logging this for now.
qDebug ( ) < < chunkComparison ( chunk , task - > summary ) ;
} else {
qWarning ( ) < < chunkComparison ( chunk , task - > summary ) ;
}
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 ;
if ( chunk - > family = = 3 & & chunk - > familyVersion = = 3 ) {
// F3V3 events are formatted as waveforms, with one chunk per mask-on slice,
// 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
//********************************************************************************************
// Internal PRS1 parsed data types
//********************************************************************************************
// For new events, add an enum here and then a class below with an PRS1_*_EVENT macro
2019-05-22 15:00:45 +00:00
enum PRS1ParsedEventType
2018-03-23 19:24:29 +00:00
{
2019-05-24 23:41:42 +00:00
EV_PRS1_RAW = - 1 , // these only get logged
2019-05-24 21:08:51 +00:00
EV_PRS1_UNKNOWN = 0 , // these have their value graphed
2019-05-22 15:00:45 +00:00
EV_PRS1_TB ,
EV_PRS1_OA ,
EV_PRS1_CA ,
EV_PRS1_FL ,
EV_PRS1_PB ,
EV_PRS1_LL ,
2019-09-19 20:43:24 +00:00
EV_PRS1_UNK_DURATION , // unknown duration event, rename once we figure it out
2019-05-22 15:00:45 +00:00
EV_PRS1_HY ,
2019-11-12 22:35:13 +00:00
EV_PRS1_OA_COUNT , // F3V3 only
EV_PRS1_CA_COUNT , // F3V3 only
EV_PRS1_HY_COUNT , // F3V3 only
2019-05-22 15:00:45 +00:00
EV_PRS1_TOTLEAK ,
2019-05-27 16:38:55 +00:00
EV_PRS1_LEAK , // unintentional leak
2019-09-19 20:43:24 +00:00
EV_PRS1_AUTO_PRESSURE_SET ,
2020-01-28 16:26:31 +00:00
EV_PRS1_PRESSURE_SET ,
2019-09-19 18:21:12 +00:00
EV_PRS1_IPAP_SET ,
EV_PRS1_EPAP_SET ,
2020-01-28 16:26:31 +00:00
EV_PRS1_PRESSURE_AVG ,
EV_PRS1_FLEX_PRESSURE_AVG ,
2019-09-19 18:21:12 +00:00
EV_PRS1_IPAP_AVG ,
2019-05-22 15:00:45 +00:00
EV_PRS1_IPAPLOW ,
EV_PRS1_IPAPHIGH ,
2019-09-19 18:21:12 +00:00
EV_PRS1_EPAP_AVG ,
2019-05-22 15:00:45 +00:00
EV_PRS1_RR ,
EV_PRS1_PTB ,
EV_PRS1_MV ,
EV_PRS1_TV ,
EV_PRS1_SNORE ,
2019-11-20 01:08:00 +00:00
EV_PRS1_VS ,
2019-05-25 21:00:44 +00:00
EV_PRS1_PP ,
2019-05-24 23:41:42 +00:00
EV_PRS1_RERA ,
2019-05-25 00:09:53 +00:00
EV_PRS1_FLOWRATE ,
2019-05-24 23:41:42 +00:00
EV_PRS1_TEST1 ,
EV_PRS1_TEST2 ,
2019-05-23 00:11:48 +00:00
EV_PRS1_SETTING ,
2019-05-26 18:17:58 +00:00
EV_PRS1_SLICE ,
2019-09-19 20:43:24 +00:00
EV_PRS1_DISCONNECT_ALARM ,
EV_PRS1_APNEA_ALARM ,
EV_PRS1_LOW_MV_ALARM ,
EV_PRS1_SNORES_AT_PRESSURE ,
2020-01-06 16:59:15 +00:00
EV_PRS1_INTERVAL_BOUNDARY , // An artificial internal-only event used to separate stat intervals
2019-05-22 15:00:45 +00:00
} ;
enum PRS1ParsedEventUnit
{
PRS1_UNIT_NONE ,
PRS1_UNIT_CMH2O ,
PRS1_UNIT_ML ,
2019-05-29 15:20:20 +00:00
PRS1_UNIT_S ,
2019-05-22 15:00:45 +00:00
} ;
2019-05-23 00:11:48 +00:00
enum PRS1ParsedSettingType
{
2019-05-25 23:21:46 +00:00
PRS1_SETTING_CPAP_MODE ,
2019-09-20 16:59:14 +00:00
PRS1_SETTING_AUTO_TRIAL ,
2019-05-25 23:43:35 +00:00
PRS1_SETTING_PRESSURE ,
PRS1_SETTING_PRESSURE_MIN ,
PRS1_SETTING_PRESSURE_MAX ,
PRS1_SETTING_EPAP ,
2019-05-23 00:11:48 +00:00
PRS1_SETTING_EPAP_MIN ,
PRS1_SETTING_EPAP_MAX ,
2019-05-25 23:43:35 +00:00
PRS1_SETTING_IPAP ,
2019-05-23 00:11:48 +00:00
PRS1_SETTING_IPAP_MIN ,
PRS1_SETTING_IPAP_MAX ,
2019-05-25 23:43:35 +00:00
PRS1_SETTING_PS ,
2019-05-23 00:11:48 +00:00
PRS1_SETTING_PS_MIN ,
PRS1_SETTING_PS_MAX ,
2019-09-23 17:43:28 +00:00
PRS1_SETTING_BACKUP_BREATH_MODE ,
PRS1_SETTING_BACKUP_BREATH_RATE ,
PRS1_SETTING_BACKUP_TIMED_INSPIRATION ,
2019-09-20 16:59:14 +00:00
PRS1_SETTING_TIDAL_VOLUME ,
PRS1_SETTING_EZ_START ,
PRS1_SETTING_FLEX_LOCK ,
2019-05-25 23:21:46 +00:00
PRS1_SETTING_FLEX_MODE ,
PRS1_SETTING_FLEX_LEVEL ,
2019-09-23 16:39:20 +00:00
PRS1_SETTING_RISE_TIME ,
2020-03-23 00:00:09 +00:00
PRS1_SETTING_RISE_TIME_LOCK ,
2019-09-20 16:59:14 +00:00
PRS1_SETTING_RAMP_TYPE ,
2019-05-25 23:21:46 +00:00
PRS1_SETTING_RAMP_TIME ,
PRS1_SETTING_RAMP_PRESSURE ,
PRS1_SETTING_HUMID_STATUS ,
2020-01-12 00:14:01 +00:00
PRS1_SETTING_HUMID_MODE ,
PRS1_SETTING_HEATED_TUBE_TEMP ,
2019-05-25 23:21:46 +00:00
PRS1_SETTING_HUMID_LEVEL ,
2019-09-20 04:15:40 +00:00
PRS1_SETTING_MASK_RESIST_LOCK ,
PRS1_SETTING_MASK_RESIST_SETTING ,
2019-05-26 01:57:29 +00:00
PRS1_SETTING_HOSE_DIAMETER ,
2020-03-23 00:00:09 +00:00
PRS1_SETTING_TUBING_LOCK ,
2019-05-26 01:57:29 +00:00
PRS1_SETTING_AUTO_ON ,
PRS1_SETTING_AUTO_OFF ,
2019-09-20 16:59:14 +00:00
PRS1_SETTING_APNEA_ALARM ,
PRS1_SETTING_DISCONNECT_ALARM , // Is this any different from mask alert?
PRS1_SETTING_LOW_MV_ALARM ,
PRS1_SETTING_LOW_TV_ALARM ,
2019-05-26 01:57:29 +00:00
PRS1_SETTING_MASK_ALERT ,
PRS1_SETTING_SHOW_AHI ,
2019-05-23 00:11:48 +00:00
} ;
2019-05-29 16:11:53 +00:00
static QString timeStr ( int t ) ;
static QString byteList ( QByteArray data , int limit = - 1 ) ;
static QString hex ( int i ) ;
static QString parsedSettingTypeName ( PRS1ParsedSettingType t ) ;
2019-08-29 01:30:25 +00:00
static QString parsedModeName ( int m ) ;
2019-05-29 16:11:53 +00:00
2019-05-22 15:00:45 +00:00
class PRS1ParsedEvent
{
public :
PRS1ParsedEventType m_type ;
int m_start ; // seconds relative to chunk timestamp at which this event began
int m_duration ;
int m_value ;
float m_offset ;
float m_gain ;
PRS1ParsedEventUnit m_unit ;
2019-05-23 00:11:48 +00:00
inline float value ( void ) { return ( m_value * m_gain ) + m_offset ; }
2019-05-28 01:02:28 +00:00
static const PRS1ParsedEventType TYPE = EV_PRS1_UNKNOWN ;
static constexpr float GAIN = 1.0 ;
static const PRS1ParsedEventUnit UNIT = PRS1_UNIT_NONE ;
2019-05-29 16:11:53 +00:00
virtual QMap < QString , QString > contents ( void ) = 0 ;
2019-05-28 01:02:28 +00:00
2019-05-22 15:00:45 +00:00
protected :
PRS1ParsedEvent ( PRS1ParsedEventType type , int start )
2019-05-28 01:02:28 +00:00
: m_type ( type ) , m_start ( start ) , m_duration ( 0 ) , m_value ( 0 ) , m_offset ( 0.0 ) , m_gain ( GAIN ) , m_unit ( UNIT )
2019-05-22 15:00:45 +00:00
{
}
2019-05-30 20:32:57 +00:00
public :
2019-05-29 16:11:53 +00:00
virtual ~ PRS1ParsedEvent ( )
2019-05-22 15:00:45 +00:00
{
}
} ;
2019-05-29 16:11:53 +00:00
2020-01-06 16:59:15 +00:00
class PRS1IntervalBoundaryEvent : public PRS1ParsedEvent
{
public :
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
return out ;
}
static const PRS1ParsedEventType TYPE = EV_PRS1_INTERVAL_BOUNDARY ;
PRS1IntervalBoundaryEvent ( int start ) : PRS1ParsedEvent ( TYPE , start ) { }
} ;
2019-05-22 15:00:45 +00:00
class PRS1ParsedDurationEvent : public PRS1ParsedEvent
2019-05-29 15:20:20 +00:00
{
2019-05-29 16:11:53 +00:00
public :
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
out [ " duration " ] = timeStr ( m_duration ) ;
return out ;
}
2019-05-29 15:20:20 +00:00
static const PRS1ParsedEventUnit UNIT = PRS1_UNIT_S ;
2019-05-28 22:25:08 +00:00
PRS1ParsedDurationEvent ( PRS1ParsedEventType type , int start , int duration ) : PRS1ParsedEvent ( type , start ) { m_duration = duration ; }
2019-05-22 15:00:45 +00:00
} ;
2019-05-29 15:20:20 +00:00
const PRS1ParsedEventUnit PRS1ParsedDurationEvent : : UNIT ;
2019-05-22 15:00:45 +00:00
2019-05-29 16:11:53 +00:00
2019-05-22 15:00:45 +00:00
class PRS1ParsedValueEvent : public PRS1ParsedEvent
{
2019-05-29 16:11:53 +00:00
public :
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
out [ " value " ] = QString : : number ( value ( ) ) ;
return out ;
}
2019-05-22 15:00:45 +00:00
protected :
PRS1ParsedValueEvent ( PRS1ParsedEventType type , int start , int value ) : PRS1ParsedEvent ( type , start ) { m_value = value ; }
} ;
2019-10-10 16:46:47 +00:00
/*
2019-05-24 21:08:51 +00:00
class PRS1UnknownValueEvent : public PRS1ParsedValueEvent
{
public :
2019-05-29 16:11:53 +00:00
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
out [ " code " ] = hex ( m_code ) ;
out [ " value " ] = QString : : number ( value ( ) ) ;
return out ;
}
2019-05-24 21:08:51 +00:00
int m_code ;
2019-05-28 01:02:28 +00:00
PRS1UnknownValueEvent ( int code , int start , int value , float gain = 1.0 ) : PRS1ParsedValueEvent ( TYPE , start , value ) , m_code ( code ) { m_gain = gain ; }
2019-05-24 21:08:51 +00:00
} ;
2019-10-10 16:46:47 +00:00
*/
2019-05-29 16:11:53 +00:00
2019-05-24 23:41:42 +00:00
class PRS1UnknownDataEvent : public PRS1ParsedEvent
{
public :
2019-05-29 16:11:53 +00:00
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " pos " ] = QString : : number ( m_pos ) ;
out [ " data " ] = byteList ( m_data ) ;
return out ;
}
2019-05-28 01:02:28 +00:00
static const PRS1ParsedEventType TYPE = EV_PRS1_RAW ;
2019-05-24 23:41:42 +00:00
int m_pos ;
unsigned char m_code ;
QByteArray m_data ;
2019-05-28 01:02:28 +00:00
2019-05-24 23:41:42 +00:00
PRS1UnknownDataEvent ( const QByteArray & data , int pos , int len = 18 )
2019-05-28 01:02:28 +00:00
: PRS1ParsedEvent ( TYPE , 0 )
2019-05-24 23:41:42 +00:00
{
m_pos = pos ;
m_data = data . mid ( pos , len ) ;
2019-06-09 01:36:51 +00:00
Q_ASSERT ( m_data . size ( ) > = 1 ) ;
2019-05-24 23:41:42 +00:00
m_code = m_data . at ( 0 ) ;
}
} ;
2019-05-22 15:00:45 +00:00
class PRS1PressureEvent : public PRS1ParsedValueEvent
{
public :
2019-05-23 00:11:48 +00:00
static constexpr float GAIN = 0.1 ;
static const PRS1ParsedEventUnit UNIT = PRS1_UNIT_CMH2O ;
2019-06-16 00:56:55 +00:00
PRS1PressureEvent ( PRS1ParsedEventType type , int start , int value , float gain = GAIN )
2019-05-22 15:00:45 +00:00
: PRS1ParsedValueEvent ( type , start , value )
{
2019-06-16 00:56:55 +00:00
m_gain = gain ;
2019-05-23 00:11:48 +00:00
m_unit = UNIT ;
}
} ;
2019-05-28 22:25:08 +00:00
class PRS1TidalVolumeEvent : public PRS1ParsedValueEvent
{
public :
static const PRS1ParsedEventType TYPE = EV_PRS1_TV ;
static constexpr float GAIN = 10.0 ;
static const PRS1ParsedEventUnit UNIT = PRS1_UNIT_ML ;
PRS1TidalVolumeEvent ( int start , int value )
: PRS1ParsedValueEvent ( TYPE , start , value )
{
m_gain = GAIN ;
m_unit = UNIT ;
}
} ;
2019-05-29 01:09:17 +00:00
const PRS1ParsedEventType PRS1TidalVolumeEvent : : TYPE ;
2019-05-28 22:25:08 +00:00
2019-05-23 00:11:48 +00:00
class PRS1ParsedSettingEvent : public PRS1ParsedValueEvent
{
public :
2019-05-29 16:11:53 +00:00
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
2019-08-29 01:30:25 +00:00
QString v ;
if ( m_setting = = PRS1_SETTING_CPAP_MODE ) {
v = parsedModeName ( value ( ) ) ;
} else {
v = QString : : number ( value ( ) ) ;
}
out [ parsedSettingTypeName ( m_setting ) ] = v ;
2019-05-29 16:11:53 +00:00
return out ;
}
2019-05-28 01:02:28 +00:00
static const PRS1ParsedEventType TYPE = EV_PRS1_SETTING ;
2019-05-23 00:11:48 +00:00
PRS1ParsedSettingType m_setting ;
2019-05-28 01:02:28 +00:00
PRS1ParsedSettingEvent ( PRS1ParsedSettingType setting , int value ) : PRS1ParsedValueEvent ( TYPE , 0 , value ) , m_setting ( setting ) { }
2019-05-23 00:11:48 +00:00
} ;
2019-09-23 17:43:28 +00:00
class PRS1ScaledSettingEvent : public PRS1ParsedSettingEvent
{
public :
PRS1ScaledSettingEvent ( PRS1ParsedSettingType setting , int value , float gain )
: PRS1ParsedSettingEvent ( setting , value )
{
m_gain = gain ;
}
} ;
class PRS1PressureSettingEvent : public PRS1ScaledSettingEvent
2019-05-23 00:11:48 +00:00
{
public :
2019-05-28 01:02:28 +00:00
static constexpr float GAIN = PRS1PressureEvent : : GAIN ;
static const PRS1ParsedEventUnit UNIT = PRS1PressureEvent : : UNIT ;
2019-06-16 00:56:55 +00:00
PRS1PressureSettingEvent ( PRS1ParsedSettingType setting , int value , float gain = GAIN )
2019-09-23 17:43:28 +00:00
: PRS1ScaledSettingEvent ( setting , value , gain )
2019-05-23 00:11:48 +00:00
{
2019-05-28 01:02:28 +00:00
m_unit = UNIT ;
2019-05-22 15:00:45 +00:00
}
} ;
2019-06-04 00:12:17 +00:00
class PRS1ParsedSliceEvent : public PRS1ParsedValueEvent
2019-05-26 18:17:58 +00:00
{
public :
2019-05-29 16:11:53 +00:00
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
QString s ;
2019-06-04 00:12:17 +00:00
switch ( ( SliceStatus ) m_value ) {
case MaskOn : s = " MaskOn " ; break ;
case MaskOff : s = " MaskOff " ; break ;
2019-05-29 16:11:53 +00:00
case EquipmentOff : s = " EquipmentOff " ; break ;
case UnknownStatus : s = " Unknown " ; break ;
}
out [ " status " ] = s ;
return out ;
}
2019-05-28 01:02:28 +00:00
static const PRS1ParsedEventType TYPE = EV_PRS1_SLICE ;
2019-06-04 00:12:17 +00:00
PRS1ParsedSliceEvent ( int start , SliceStatus status ) : PRS1ParsedValueEvent ( TYPE , start , ( int ) status ) { }
2019-05-28 22:25:08 +00:00
} ;
2019-09-19 20:43:24 +00:00
class PRS1ParsedAlarmEvent : public PRS1ParsedEvent
{
public :
virtual QMap < QString , QString > contents ( void )
{
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
return out ;
}
protected :
PRS1ParsedAlarmEvent ( PRS1ParsedEventType type , int start , int /*unused*/ ) : PRS1ParsedEvent ( type , start ) { }
} ;
class PRS1SnoresAtPressureEvent : public PRS1PressureEvent
{
public :
static const PRS1ParsedEventType TYPE = EV_PRS1_SNORES_AT_PRESSURE ;
PRS1SnoresAtPressureEvent ( int start , int kind , int pressure , int count , float gain = GAIN )
: PRS1PressureEvent ( TYPE , start , pressure , gain )
{
m_kind = kind ;
m_count = count ;
}
virtual QMap < QString , QString > contents ( void )
{
QString label ;
switch ( m_kind ) {
case 0 : label = " pressure " ; break ;
case 1 : label = " epap " ; break ;
case 2 : label = " ipap " ; break ;
default : label = " unknown_pressure " ; break ;
}
QMap < QString , QString > out ;
out [ " start " ] = timeStr ( m_start ) ;
out [ label ] = QString : : number ( value ( ) ) ;
out [ " count " ] = QString : : number ( m_count ) ;
return out ;
}
protected :
int m_kind ;
// m_value is pressure
int m_count ;
} ;
const PRS1ParsedEventType PRS1SnoresAtPressureEvent : : TYPE ;
2019-05-28 01:02:28 +00:00
# define _PRS1_EVENT(T, E, P, ARG) \
class T : public P \
{ \
public : \
static const PRS1ParsedEventType TYPE = E ; \
T ( int start , int ARG ) : P ( TYPE , start , ARG ) { } \
2019-05-29 01:09:17 +00:00
} ; \
const PRS1ParsedEventType T : : TYPE
2019-05-28 01:02:28 +00:00
# define PRS1_DURATION_EVENT(T, E) _PRS1_EVENT(T, E, PRS1ParsedDurationEvent, duration)
# define PRS1_VALUE_EVENT(T, E) _PRS1_EVENT(T, E, PRS1ParsedValueEvent, value)
2019-09-19 20:43:24 +00:00
# define PRS1_ALARM_EVENT(T, E) _PRS1_EVENT(T, E, PRS1ParsedAlarmEvent, value)
2019-06-16 00:56:55 +00:00
# define PRS1_PRESSURE_EVENT(T, E) \
class T : public PRS1PressureEvent \
{ \
public : \
static const PRS1ParsedEventType TYPE = E ; \
T ( int start , int value , float gain = PRS1PressureEvent : : GAIN ) : PRS1PressureEvent ( TYPE , start , value , gain ) { } \
} ; \
const PRS1ParsedEventType T : : TYPE
2019-05-26 18:17:58 +00:00
2019-05-28 01:02:28 +00:00
PRS1_DURATION_EVENT ( PRS1TimedBreathEvent , EV_PRS1_TB ) ;
PRS1_DURATION_EVENT ( PRS1ObstructiveApneaEvent , EV_PRS1_OA ) ;
PRS1_DURATION_EVENT ( PRS1ClearAirwayEvent , EV_PRS1_CA ) ;
PRS1_DURATION_EVENT ( PRS1FlowLimitationEvent , EV_PRS1_FL ) ;
PRS1_DURATION_EVENT ( PRS1PeriodicBreathingEvent , EV_PRS1_PB ) ;
PRS1_DURATION_EVENT ( PRS1LargeLeakEvent , EV_PRS1_LL ) ;
2019-09-19 20:43:24 +00:00
PRS1_DURATION_EVENT ( PRS1UnknownDurationEvent , EV_PRS1_UNK_DURATION ) ;
2019-05-28 01:02:28 +00:00
PRS1_DURATION_EVENT ( PRS1HypopneaEvent , EV_PRS1_HY ) ;
2019-05-22 15:00:45 +00:00
2019-05-28 01:02:28 +00:00
PRS1_VALUE_EVENT ( PRS1TotalLeakEvent , EV_PRS1_TOTLEAK ) ;
2019-10-24 16:19:13 +00:00
PRS1_VALUE_EVENT ( PRS1LeakEvent , EV_PRS1_LEAK ) ;
2019-05-24 21:08:51 +00:00
2019-09-19 20:43:24 +00:00
PRS1_PRESSURE_EVENT ( PRS1AutoPressureSetEvent , EV_PRS1_AUTO_PRESSURE_SET ) ;
2019-09-19 18:21:12 +00:00
PRS1_PRESSURE_EVENT ( PRS1PressureSetEvent , EV_PRS1_PRESSURE_SET ) ;
PRS1_PRESSURE_EVENT ( PRS1IPAPSetEvent , EV_PRS1_IPAP_SET ) ;
PRS1_PRESSURE_EVENT ( PRS1EPAPSetEvent , EV_PRS1_EPAP_SET ) ;
PRS1_PRESSURE_EVENT ( PRS1PressureAverageEvent , EV_PRS1_PRESSURE_AVG ) ;
2020-01-28 16:26:31 +00:00
PRS1_PRESSURE_EVENT ( PRS1FlexPressureAverageEvent , EV_PRS1_FLEX_PRESSURE_AVG ) ;
2019-09-19 18:21:12 +00:00
PRS1_PRESSURE_EVENT ( PRS1IPAPAverageEvent , EV_PRS1_IPAP_AVG ) ;
2019-05-28 01:02:28 +00:00
PRS1_PRESSURE_EVENT ( PRS1IPAPHighEvent , EV_PRS1_IPAPHIGH ) ;
PRS1_PRESSURE_EVENT ( PRS1IPAPLowEvent , EV_PRS1_IPAPLOW ) ;
2019-09-19 18:21:12 +00:00
PRS1_PRESSURE_EVENT ( PRS1EPAPAverageEvent , EV_PRS1_EPAP_AVG ) ;
2019-05-25 21:00:44 +00:00
2019-05-28 01:02:28 +00:00
PRS1_VALUE_EVENT ( PRS1RespiratoryRateEvent , EV_PRS1_RR ) ;
PRS1_VALUE_EVENT ( PRS1PatientTriggeredBreathsEvent , EV_PRS1_PTB ) ;
PRS1_VALUE_EVENT ( PRS1MinuteVentilationEvent , EV_PRS1_MV ) ;
PRS1_VALUE_EVENT ( PRS1SnoreEvent , EV_PRS1_SNORE ) ;
PRS1_VALUE_EVENT ( PRS1VibratorySnoreEvent , EV_PRS1_VS ) ;
PRS1_VALUE_EVENT ( PRS1PressurePulseEvent , EV_PRS1_PP ) ;
PRS1_VALUE_EVENT ( PRS1RERAEvent , EV_PRS1_RERA ) ; // TODO: should this really be a duration event?
PRS1_VALUE_EVENT ( PRS1FlowRateEvent , EV_PRS1_FLOWRATE ) ; // TODO: is this a single event or an index/hour?
2019-05-29 01:09:17 +00:00
PRS1_VALUE_EVENT ( PRS1Test1Event , EV_PRS1_TEST1 ) ;
2019-05-28 01:02:28 +00:00
PRS1_VALUE_EVENT ( PRS1Test2Event , EV_PRS1_TEST2 ) ;
2019-11-12 22:35:13 +00:00
PRS1_VALUE_EVENT ( PRS1HypopneaCount , EV_PRS1_HY_COUNT ) ; // F3V3 only
PRS1_VALUE_EVENT ( PRS1ClearAirwayCount , EV_PRS1_CA_COUNT ) ; // F3V3 only
PRS1_VALUE_EVENT ( PRS1ObstructiveApneaCount , EV_PRS1_OA_COUNT ) ; // F3V3 only
2019-05-24 23:41:42 +00:00
2019-09-19 20:43:24 +00:00
PRS1_ALARM_EVENT ( PRS1DisconnectAlarmEvent , EV_PRS1_DISCONNECT_ALARM ) ;
PRS1_ALARM_EVENT ( PRS1ApneaAlarmEvent , EV_PRS1_APNEA_ALARM ) ;
PRS1_ALARM_EVENT ( PRS1LowMinuteVentilationAlarmEvent , EV_PRS1_LOW_MV_ALARM ) ;
2019-05-29 16:11:53 +00:00
2019-08-29 01:30:25 +00:00
enum PRS1Mode {
PRS1_MODE_UNKNOWN = - 1 ,
PRS1_MODE_CPAP = 0 , // "CPAP"
PRS1_MODE_CPAPCHECK , // "CPAP-Check"
PRS1_MODE_AUTOCPAP , // "AutoCPAP"
PRS1_MODE_BILEVEL , // "Bi-Level"
PRS1_MODE_AUTOBILEVEL , // "AutoBiLevel"
PRS1_MODE_ASV , // "ASV"
PRS1_MODE_S , // "S"
PRS1_MODE_ST , // "S/T"
PRS1_MODE_PC , // "PC"
2019-09-26 16:37:01 +00:00
PRS1_MODE_AUTOTRIAL , // "Auto-Trial"
2019-08-29 01:30:25 +00:00
} ;
2019-10-20 21:46:18 +00:00
// Returns the set of all channels ever reported/supported by the parser for the given chunk.
2019-10-19 21:35:45 +00:00
const QVector < PRS1ParsedEventType > & GetSupportedEvents ( const PRS1DataChunk * chunk ) ;
2019-08-29 01:30:25 +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 =
{
//PRS1TimedBreathEvent::TYPE, // TODO: TB could be on-demand
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 } } ,
2019-10-22 16:30:42 +00:00
{ PRS1TotalLeakEvent : : TYPE , { & CPAP_LeakTotal , & CPAP_Leak } } , // TODO: Remove CPAP_Leak if we get rid of unintentional leak calculation in the importer.
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 } } ,
2019-10-21 01:09:34 +00:00
{ PRS1FlowRateEvent : : TYPE , { & CPAP_FlowRate } } , // Only reported by 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 */ } } ,
2019-10-29 18:06:57 +00:00
{ PRS1UnknownDurationEvent : : TYPE , { & PRS1_0E } } ,
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
//********************************************************************************************
static QString hex ( int i )
{
return QString ( " 0x " ) + QString : : number ( i , 16 ) . toUpper ( ) ;
}
# define ENUMSTRING(ENUM) case ENUM: s = #ENUM; break
static QString parsedEventTypeName ( PRS1ParsedEventType t )
{
QString s ;
switch ( t ) {
ENUMSTRING ( EV_PRS1_RAW ) ;
ENUMSTRING ( EV_PRS1_UNKNOWN ) ;
ENUMSTRING ( EV_PRS1_TB ) ;
ENUMSTRING ( EV_PRS1_OA ) ;
ENUMSTRING ( EV_PRS1_CA ) ;
ENUMSTRING ( EV_PRS1_FL ) ;
ENUMSTRING ( EV_PRS1_PB ) ;
ENUMSTRING ( EV_PRS1_LL ) ;
2019-09-19 20:43:24 +00:00
ENUMSTRING ( EV_PRS1_UNK_DURATION ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( EV_PRS1_HY ) ;
2019-11-20 01:08:00 +00:00
ENUMSTRING ( EV_PRS1_OA_COUNT ) ;
ENUMSTRING ( EV_PRS1_CA_COUNT ) ;
ENUMSTRING ( EV_PRS1_HY_COUNT ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( EV_PRS1_TOTLEAK ) ;
ENUMSTRING ( EV_PRS1_LEAK ) ;
2019-09-19 20:43:24 +00:00
ENUMSTRING ( EV_PRS1_AUTO_PRESSURE_SET ) ;
2019-09-19 18:21:12 +00:00
ENUMSTRING ( EV_PRS1_PRESSURE_SET ) ;
ENUMSTRING ( EV_PRS1_IPAP_SET ) ;
ENUMSTRING ( EV_PRS1_EPAP_SET ) ;
ENUMSTRING ( EV_PRS1_PRESSURE_AVG ) ;
2020-01-28 16:26:31 +00:00
ENUMSTRING ( EV_PRS1_FLEX_PRESSURE_AVG ) ;
2019-09-19 18:21:12 +00:00
ENUMSTRING ( EV_PRS1_IPAP_AVG ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( EV_PRS1_IPAPLOW ) ;
ENUMSTRING ( EV_PRS1_IPAPHIGH ) ;
2019-09-19 18:21:12 +00:00
ENUMSTRING ( EV_PRS1_EPAP_AVG ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( EV_PRS1_RR ) ;
ENUMSTRING ( EV_PRS1_PTB ) ;
ENUMSTRING ( EV_PRS1_MV ) ;
ENUMSTRING ( EV_PRS1_TV ) ;
ENUMSTRING ( EV_PRS1_SNORE ) ;
ENUMSTRING ( EV_PRS1_VS ) ;
ENUMSTRING ( EV_PRS1_PP ) ;
ENUMSTRING ( EV_PRS1_RERA ) ;
ENUMSTRING ( EV_PRS1_FLOWRATE ) ;
ENUMSTRING ( EV_PRS1_TEST1 ) ;
ENUMSTRING ( EV_PRS1_TEST2 ) ;
ENUMSTRING ( EV_PRS1_SETTING ) ;
ENUMSTRING ( EV_PRS1_SLICE ) ;
2019-09-19 20:43:24 +00:00
ENUMSTRING ( EV_PRS1_DISCONNECT_ALARM ) ;
ENUMSTRING ( EV_PRS1_APNEA_ALARM ) ;
ENUMSTRING ( EV_PRS1_LOW_MV_ALARM ) ;
ENUMSTRING ( EV_PRS1_SNORES_AT_PRESSURE ) ;
2020-01-06 16:59:15 +00:00
ENUMSTRING ( EV_PRS1_INTERVAL_BOUNDARY ) ;
2019-05-29 16:11:53 +00:00
default :
s = hex ( t ) ;
qDebug ( ) < < " Unknown PRS1ParsedEventType type: " < < qPrintable ( s ) ;
return s ;
}
return s . mid ( 8 ) . toLower ( ) ; // lop off initial EV_PRS1_
}
static QString parsedSettingTypeName ( PRS1ParsedSettingType t )
{
QString s ;
switch ( t ) {
ENUMSTRING ( PRS1_SETTING_CPAP_MODE ) ;
2019-09-20 16:59:14 +00:00
ENUMSTRING ( PRS1_SETTING_AUTO_TRIAL ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_PRESSURE ) ;
ENUMSTRING ( PRS1_SETTING_PRESSURE_MIN ) ;
ENUMSTRING ( PRS1_SETTING_PRESSURE_MAX ) ;
ENUMSTRING ( PRS1_SETTING_EPAP ) ;
ENUMSTRING ( PRS1_SETTING_EPAP_MIN ) ;
ENUMSTRING ( PRS1_SETTING_EPAP_MAX ) ;
ENUMSTRING ( PRS1_SETTING_IPAP ) ;
ENUMSTRING ( PRS1_SETTING_IPAP_MIN ) ;
ENUMSTRING ( PRS1_SETTING_IPAP_MAX ) ;
ENUMSTRING ( PRS1_SETTING_PS ) ;
ENUMSTRING ( PRS1_SETTING_PS_MIN ) ;
ENUMSTRING ( PRS1_SETTING_PS_MAX ) ;
2019-09-23 17:43:28 +00:00
ENUMSTRING ( PRS1_SETTING_BACKUP_BREATH_MODE ) ;
ENUMSTRING ( PRS1_SETTING_BACKUP_BREATH_RATE ) ;
ENUMSTRING ( PRS1_SETTING_BACKUP_TIMED_INSPIRATION ) ;
2019-09-20 16:59:14 +00:00
ENUMSTRING ( PRS1_SETTING_TIDAL_VOLUME ) ;
ENUMSTRING ( PRS1_SETTING_EZ_START ) ;
ENUMSTRING ( PRS1_SETTING_FLEX_LOCK ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_FLEX_MODE ) ;
ENUMSTRING ( PRS1_SETTING_FLEX_LEVEL ) ;
2019-09-23 16:39:20 +00:00
ENUMSTRING ( PRS1_SETTING_RISE_TIME ) ;
2020-03-23 00:00:09 +00:00
ENUMSTRING ( PRS1_SETTING_RISE_TIME_LOCK ) ;
2019-09-20 16:59:14 +00:00
ENUMSTRING ( PRS1_SETTING_RAMP_TYPE ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_RAMP_TIME ) ;
ENUMSTRING ( PRS1_SETTING_RAMP_PRESSURE ) ;
ENUMSTRING ( PRS1_SETTING_HUMID_STATUS ) ;
2020-01-12 00:14:01 +00:00
ENUMSTRING ( PRS1_SETTING_HUMID_MODE ) ;
ENUMSTRING ( PRS1_SETTING_HEATED_TUBE_TEMP ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_HUMID_LEVEL ) ;
2019-09-20 04:15:40 +00:00
ENUMSTRING ( PRS1_SETTING_MASK_RESIST_LOCK ) ;
ENUMSTRING ( PRS1_SETTING_MASK_RESIST_SETTING ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_HOSE_DIAMETER ) ;
2020-03-23 00:00:09 +00:00
ENUMSTRING ( PRS1_SETTING_TUBING_LOCK ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_AUTO_ON ) ;
ENUMSTRING ( PRS1_SETTING_AUTO_OFF ) ;
2019-09-20 16:59:14 +00:00
ENUMSTRING ( PRS1_SETTING_APNEA_ALARM ) ;
ENUMSTRING ( PRS1_SETTING_DISCONNECT_ALARM ) ;
ENUMSTRING ( PRS1_SETTING_LOW_MV_ALARM ) ;
ENUMSTRING ( PRS1_SETTING_LOW_TV_ALARM ) ;
2019-05-29 16:11:53 +00:00
ENUMSTRING ( PRS1_SETTING_MASK_ALERT ) ;
ENUMSTRING ( PRS1_SETTING_SHOW_AHI ) ;
default :
s = hex ( t ) ;
qDebug ( ) < < " Unknown PRS1ParsedSettingType type: " < < qPrintable ( s ) ;
return s ;
}
return s . mid ( 13 ) . toLower ( ) ; // lop off initial PRS1_SETTING_
}
2019-08-29 01:30:25 +00:00
static QString parsedModeName ( int m )
{
QString s ;
switch ( ( PRS1Mode ) m ) {
ENUMSTRING ( PRS1_MODE_UNKNOWN ) ; // TODO: Remove this when all the parsers are complete.
ENUMSTRING ( PRS1_MODE_CPAP ) ;
ENUMSTRING ( PRS1_MODE_CPAPCHECK ) ;
2019-09-26 16:37:01 +00:00
ENUMSTRING ( PRS1_MODE_AUTOTRIAL ) ;
2019-08-29 01:30:25 +00:00
ENUMSTRING ( PRS1_MODE_AUTOCPAP ) ;
ENUMSTRING ( PRS1_MODE_BILEVEL ) ;
ENUMSTRING ( PRS1_MODE_AUTOBILEVEL ) ;
ENUMSTRING ( PRS1_MODE_ASV ) ;
ENUMSTRING ( PRS1_MODE_S ) ;
ENUMSTRING ( PRS1_MODE_ST ) ;
ENUMSTRING ( PRS1_MODE_PC ) ;
default :
s = hex ( m ) ;
qDebug ( ) < < " Unknown PRS1Mode: " < < qPrintable ( s ) ;
return s ;
}
return s . mid ( 10 ) . toLower ( ) ; // lop off initial PRS1_MODE_
}
2019-05-29 16:11:53 +00:00
static QString timeStr ( int t )
{
int h = t / 3600 ;
int m = ( t - ( h * 3600 ) ) / 60 ;
int s = t % 60 ;
return QString ( " %1:%2:%3 " ) . arg ( h , 2 , 10 , QChar ( ' 0 ' ) ) . arg ( m , 2 , 10 , QChar ( ' 0 ' ) ) . arg ( s , 2 , 10 , QChar ( ' 0 ' ) ) ;
}
static QString byteList ( QByteArray data , int limit )
{
int count = data . size ( ) ;
if ( limit = = - 1 | | limit > count ) limit = count ;
QStringList l ;
for ( int i = 0 ; i < limit ; i + + ) {
l . push_back ( QString ( " %1 " ) . arg ( ( int ) data [ i ] & 0xFF , 2 , 16 , QChar ( ' 0 ' ) ) . toUpper ( ) ) ;
}
if ( limit < count ) l . push_back ( " ... " ) ;
QString s = l . join ( " " ) ;
return s ;
}
QString _PRS1ParsedEventName ( PRS1ParsedEvent * e )
{
return parsedEventTypeName ( e - > m_type ) ;
}
QMap < QString , QString > _PRS1ParsedEventContents ( PRS1ParsedEvent * e )
{
return e - > contents ( ) ;
}
2019-05-28 22:25:08 +00:00
//********************************************************************************************
2019-10-04 23:51:34 +00:00
static QString DumpEvent ( int t , int code , const unsigned char * data , int size )
{
int s = t ;
int h = s / 3600 ; s - = h * 3600 ;
int m = s / 60 ; s - = m * 60 ;
2019-10-06 00:52:34 +00:00
QString dump = QString ( " %1:%2:%3 " )
2019-10-04 23:51:34 +00:00
. arg ( h , 2 , 10 , QChar ( ' 0 ' ) )
. arg ( m , 2 , 10 , QChar ( ' 0 ' ) )
. arg ( s , 2 , 10 , QChar ( ' 0 ' ) ) ;
dump = dump + " " + hex ( code ) + " : " ;
for ( int i = 0 ; i < size ; i + + ) {
dump = dump + QString ( " %1 " ) . arg ( data [ i ] ) ;
}
return dump ;
}
2019-10-06 00:52:34 +00:00
# define DUMP_EVENT() qWarning() << this->sessionid << DumpEvent(t, code, data + pos, size - (pos - startpos)) + " @ " + hex(startpos-1)
2019-10-04 23:51:34 +00:00
2019-05-22 15:00:45 +00:00
void PRS1DataChunk : : AddEvent ( PRS1ParsedEvent * const event )
{
m_parsedData . push_back ( event ) ;
}
2019-10-22 14:24:40 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF5V3 = {
PRS1EPAPSetEvent : : TYPE ,
PRS1TimedBreathEvent : : TYPE ,
PRS1IPAPAverageEvent : : TYPE ,
PRS1IPAPLowEvent : : TYPE ,
PRS1IPAPHighEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1RespiratoryRateEvent : : TYPE ,
PRS1PatientTriggeredBreathsEvent : : TYPE ,
PRS1MinuteVentilationEvent : : TYPE ,
PRS1TidalVolumeEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
PRS1EPAPAverageEvent : : TYPE ,
PRS1LeakEvent : : TYPE ,
PRS1PressurePulseEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
PRS1VibratorySnoreEvent : : TYPE ,
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
} ;
2019-06-16 00:56:55 +00:00
// Outer loop based on ParseSummaryF5V3 along with hint as to event codes from old ParseEventsF5V3,
// except this actually does something with the data.
bool PRS1DataChunk : : ParseEventsF5V3 ( void )
{
if ( this - > family ! = 5 | | this - > familyVersion ! = 3 ) {
qWarning ( ) < < " ParseEventsF5V3 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
static const int minimum_sizes [ ] = { 2 , 3 , 3 , 0xd , 3 , 3 , 3 , 4 , 3 , 2 , 5 , 5 , 3 , 3 , 3 , 3 } ;
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
if ( chunk_size < 1 ) {
// This does occasionally happen.
2019-06-19 20:23:28 +00:00
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
2019-06-16 00:56:55 +00:00
return false ;
}
// F5V3 uses a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
static const float GAIN = 0.125 ; // TODO: this should be parameterized somewhere more logical
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
int elapsed , duration ;
do {
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for event " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
if ( size < minimum_sizes [ code ] ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " too small " < < size < < " < " < < minimum_sizes [ code ] ;
ok = false ;
break ;
}
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
pos + = 2 ;
switch ( code ) {
case 1 : // Pressure adjustment
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos + + ] , GAIN ) ) ;
2019-10-08 20:42:47 +00:00
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos - 1 , size + 1 ) ) ; // TODO: what is this?
2019-06-16 00:56:55 +00:00
break ;
case 2 : // Timed Breath
2019-06-17 21:33:39 +00:00
// TB events have a duration in 0.1s, based on the review of pressure waveforms.
// TODO: Ideally the starting time here would be adjusted here, but PRS1ParsedEvents
// currently assume integer seconds rather than ms, so that's done at import.
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ;
2019-06-19 20:23:28 +00:00
this - > AddEvent ( new PRS1TimedBreathEvent ( t , duration ) ) ;
2019-06-16 00:56:55 +00:00
break ;
case 3 : // Statistics
// These appear every 2 minutes, so presumably summarize the preceding period.
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1IPAPAverageEvent ( t , data [ pos + 0 ] , GAIN ) ) ; // 00=IPAP
2019-08-20 15:38:55 +00:00
this - > AddEvent ( new PRS1IPAPLowEvent ( t , data [ pos + 1 ] , GAIN ) ) ; // 01=IAP Low
this - > AddEvent ( new PRS1IPAPHighEvent ( t , data [ pos + 2 ] , GAIN ) ) ; // 02=IAP High
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos + 3 ] ) ) ; // 03=Total leak (average?)
this - > AddEvent ( new PRS1RespiratoryRateEvent ( t , data [ pos + 4 ] ) ) ; // 04=Breaths Per Minute (average?)
this - > AddEvent ( new PRS1PatientTriggeredBreathsEvent ( t , data [ pos + 5 ] ) ) ; // 05=Patient Triggered Breaths (average?)
this - > AddEvent ( new PRS1MinuteVentilationEvent ( t , data [ pos + 6 ] ) ) ; // 06=Minute Ventilation (average?)
this - > AddEvent ( new PRS1TidalVolumeEvent ( t , data [ pos + 7 ] ) ) ; // 07=Tidal Volume (average?)
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 8 ] ) ) ; // 08=Snore count // TODO: not a VS on official waveform, but appears in flags and contributes to overall VS index
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1EPAPAverageEvent ( t , data [ pos + 9 ] , GAIN ) ) ; // 09=EPAP average
2019-08-20 15:38:55 +00:00
this - > AddEvent ( new PRS1LeakEvent ( t , data [ pos + 0xa ] ) ) ; // 0A=Leak (average?)
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-06-16 00:56:55 +00:00
break ;
2019-06-19 20:23:28 +00:00
case 0x04 : // Pressure Pulse
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ; // TODO: is this a duration?
2019-06-19 20:23:28 +00:00
this - > AddEvent ( new PRS1PressurePulseEvent ( t , duration ) ) ;
break ;
2019-06-16 00:56:55 +00:00
case 0x05 : // Obstructive Apnea
2019-06-17 21:33:39 +00:00
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-06-16 00:56:55 +00:00
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x06 : // Clear Airway Apnea
2019-06-17 21:33:39 +00:00
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-06-16 00:56:55 +00:00
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
2019-06-19 20:23:28 +00:00
case 0x07 : // Hypopnea
// TODO: How is this hypopnea different from events 0xd and 0xe?
// TODO: What is the first byte?
2019-08-20 15:38:55 +00:00
//data[pos+0]; // unknown first byte?
elapsed = data [ pos + 1 ] ; // based on sample waveform, the hypopnea is over after this
2019-06-19 20:23:28 +00:00
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
2019-06-16 00:56:55 +00:00
case 0x08 : // Flow Limitation
2019-06-19 20:23:28 +00:00
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-06-19 20:23:28 +00:00
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
2019-06-16 00:56:55 +00:00
break ;
case 0x09 : // Vibratory Snore
2019-06-17 21:33:39 +00:00
// VS events are instantaneous flags with no duration, drawn on the official waveform.
// The current thinking is that these are the snores that cause a change in auto-titrating
// pressure. The snoring statistic above seems to be a total count. It's unclear whether
// the trigger for pressure change is severity or count or something else.
2019-06-16 00:56:55 +00:00
// no data bytes
2019-06-17 21:33:39 +00:00
this - > AddEvent ( new PRS1VibratorySnoreEvent ( t , 0 ) ) ;
2019-06-16 00:56:55 +00:00
break ;
case 0x0a : // Periodic Breathing
2019-06-17 21:33:39 +00:00
// PB events are reported some time after they conclude, and they do have a reported duration.
2019-06-16 00:56:55 +00:00
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ;
2019-06-16 01:36:58 +00:00
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
2019-06-16 00:56:55 +00:00
break ;
case 0x0b : // Large Leak
2019-06-17 21:33:39 +00:00
// LL events are reported some time after they conclude, and they do have a reported duration.
2019-06-16 00:56:55 +00:00
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ;
2019-06-16 01:36:58 +00:00
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
2019-06-16 00:56:55 +00:00
break ;
2019-06-19 20:23:28 +00:00
case 0x0d : // Hypopnea
// TODO: Why does this hypopnea have a different event code?
// fall through
2019-06-16 00:56:55 +00:00
case 0x0e : // Hypopnea
2019-06-19 20:23:28 +00:00
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating hypopneas ourselves. Their official definition
// is 40% reduction in flow lasting at least 10s.
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ;
2019-06-19 20:23:28 +00:00
this - > AddEvent ( new PRS1HypopneaEvent ( t - duration , 0 ) ) ;
break ;
case 0x0f :
// TODO: some other pressure adjustment?
// Appears near the beginning and end of a session when Opti-Start is on, at least once in middle
//CHECK_VALUES(data[pos], 0x20, 0x28);
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos - 1 , size + 1 ) ) ;
2019-06-16 00:56:55 +00:00
break ;
default :
2020-03-23 00:00:09 +00:00
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
2019-06-16 00:56:55 +00:00
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos - 1 , size + 1 ) ) ;
break ;
}
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
this - > duration = t ;
return ok ;
}
2018-03-23 19:24:29 +00:00
2019-10-22 14:24:40 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF5V0 = {
PRS1EPAPSetEvent : : TYPE ,
PRS1TimedBreathEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
PRS1VibratorySnoreEvent : : TYPE ,
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
PRS1IPAPAverageEvent : : TYPE ,
PRS1IPAPLowEvent : : TYPE ,
PRS1IPAPHighEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1RespiratoryRateEvent : : TYPE ,
PRS1PatientTriggeredBreathsEvent : : TYPE ,
PRS1MinuteVentilationEvent : : TYPE ,
PRS1TidalVolumeEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
PRS1EPAPAverageEvent : : TYPE ,
} ;
2019-10-05 00:10:35 +00:00
// 950P is F5V0
bool PRS1DataChunk : : ParseEventsF5V0 ( void )
2019-05-24 21:08:51 +00:00
{
2019-10-05 00:10:35 +00:00
if ( this - > family ! = 5 | | this - > familyVersion ! = 0 ) {
qWarning ( ) < < " ParseEventsF5V0 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
2019-10-04 23:51:34 +00:00
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2019-11-20 17:33:45 +00:00
static const QMap < int , int > event_sizes = { { 1 , 2 } , { 3 , 4 } , { 8 , 4 } , { 0xa , 2 } , { 0xb , 5 } , { 0xc , 5 } , { 0xd , 0xc } } ;
2019-10-05 00:10:35 +00:00
if ( chunk_size < 1 ) {
// This does occasionally happen in F0V6.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
2019-10-06 00:52:34 +00:00
int elapsed , duration ;
2019-10-05 00:10:35 +00:00
do {
code = data [ pos + + ] ;
size = 3 ; // default size = 2 bytes time delta + 1 byte data
if ( event_sizes . contains ( code ) ) {
size = event_sizes [ code ] ;
}
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
2019-10-06 00:52:34 +00:00
if ( code ! = 0 ) { // Does this code really not have a timestamp? See below where we check.
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2019-10-05 00:10:35 +00:00
pos + = 2 ;
}
switch ( code ) {
2019-10-06 00:52:34 +00:00
case 0x00 : // Unknown, only seen twice
//DUMP_EVENT();
// So far we've only seen 0 for the first 2 bytes. Look for nonzero to see if it's actually a timestamp. If so, fix above to read it.
CHECK_VALUE ( data [ pos ] , 0 ) ;
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ;
2019-11-20 17:33:45 +00:00
if ( data [ pos + 2 ] ! = 0x85 ) {
CHECK_VALUES ( data [ pos + 2 ] , 0x81 , 0x83 ) ; // Only three values seen so far
}
2019-10-06 00:52:34 +00:00
break ;
//case 0x01: // never seen on F5V0
case 0x02 : // Pressure adjustment
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos + + ] ) ) ;
break ;
2019-12-01 23:21:18 +00:00
//case 0x03: // never seen on F5V0; probably pressure pulse, see F5V1
2019-10-06 00:52:34 +00:00
case 0x04 : // Timed Breath
// TB events have a duration in 0.1s, based on the review of pressure waveforms.
// TODO: Ideally the starting time here would be adjusted here, but PRS1ParsedEvents
// currently assume integer seconds rather than ms, so that's done at import.
duration = data [ pos ] ;
this - > AddEvent ( new PRS1TimedBreathEvent ( t , duration ) ) ;
break ;
case 0x05 : // Obstructive Apnea
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x06 : // Clear Airway Apnea
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x07 : // Hypopnea
2019-10-08 14:44:18 +00:00
// NOTE: No additional (unknown) first byte as in F5V3 0x07, but see below.
// This seems closer to F5V3 0x0d or 0x0e.
2019-10-06 00:52:34 +00:00
elapsed = data [ pos ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
2019-11-20 17:33:45 +00:00
case 0x08 : // Hypopnea? See F5V1
// This has similar structure fo the event 8 HY in F5V1, but it doesn't seem
// to be drawn on official reports. This has been seen only once, along with
// a simultaneous event 7 HY, and only one HY was drawn. (This would have
// started at 2:43:47, and the subsequent HY starts at 2:43:51.)
//
// The event length at least is confirmed, so we can keep parsing, but
// for now don't import and just alert us to any other examples of this event.
CHECK_VALUE ( t , 9860 ) ;
CHECK_VALUE ( sessionid , 484 ) ;
break ;
2019-10-06 00:52:34 +00:00
case 0x09 : // Flow Limitation, note this is 0x8 in F5V3
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0a : // Vibratory Snore, note this is 0x9 in F5V3
// VS events are instantaneous flags with no duration, drawn on the official waveform.
// The current thinking is that these are the snores that cause a change in auto-titrating
// pressure. The snoring statistic above seems to be a total count. It's unclear whether
// the trigger for pressure change is severity or count or something else.
// no data bytes
this - > AddEvent ( new PRS1VibratorySnoreEvent ( t , 0 ) ) ;
break ;
case 0x0b : // Periodic Breathing, note this is 0xa in F5V3
// PB events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // confirmed to double in F5V0
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x0c : // Large Leak, note this is 0xb in F5V3
// LL events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // confirmed to double in F5V0
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x0d : // Statistics
// These appear every 2 minutes, so presumably summarize the preceding period.
this - > AddEvent ( new PRS1IPAPAverageEvent ( t , data [ pos + 0 ] ) ) ; // 00=IPAP
this - > AddEvent ( new PRS1IPAPLowEvent ( t , data [ pos + 1 ] ) ) ; // 01=IAP Low
this - > AddEvent ( new PRS1IPAPHighEvent ( t , data [ pos + 2 ] ) ) ; // 02=IAP High
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos + 3 ] ) ) ; // 03=Total leak (average?)
this - > AddEvent ( new PRS1RespiratoryRateEvent ( t , data [ pos + 4 ] ) ) ; // 04=Breaths Per Minute (average?)
this - > AddEvent ( new PRS1PatientTriggeredBreathsEvent ( t , data [ pos + 5 ] ) ) ; // 05=Patient Triggered Breaths (average?)
this - > AddEvent ( new PRS1MinuteVentilationEvent ( t , data [ pos + 6 ] ) ) ; // 06=Minute Ventilation (average?)
this - > AddEvent ( new PRS1TidalVolumeEvent ( t , data [ pos + 7 ] ) ) ; // 07=Tidal Volume (average?)
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 8 ] ) ) ; // 08=Snore count // TODO: not a VS on official waveform, but appears in flags and contributes to overall VS index
this - > AddEvent ( new PRS1EPAPAverageEvent ( t , data [ pos + 9 ] ) ) ; // 09=EPAP average
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-10-06 00:52:34 +00:00
break ;
2019-10-05 00:10:35 +00:00
default :
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos ) ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
}
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing event bytes " ;
}
this - > duration = t ;
return ok ;
}
2019-10-22 14:24:40 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF5V1 = {
PRS1EPAPSetEvent : : TYPE ,
2019-12-01 23:21:18 +00:00
PRS1PressurePulseEvent : : TYPE ,
2019-10-22 14:24:40 +00:00
PRS1TimedBreathEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
PRS1VibratorySnoreEvent : : TYPE ,
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
PRS1IPAPAverageEvent : : TYPE ,
PRS1IPAPLowEvent : : TYPE ,
PRS1IPAPHighEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1RespiratoryRateEvent : : TYPE ,
PRS1PatientTriggeredBreathsEvent : : TYPE ,
PRS1MinuteVentilationEvent : : TYPE ,
PRS1TidalVolumeEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
PRS1EPAPAverageEvent : : TYPE ,
PRS1LeakEvent : : TYPE ,
} ;
2019-10-05 00:10:35 +00:00
// 960P and 961P are F5V1
bool PRS1DataChunk : : ParseEventsF5V1 ( void )
{
if ( this - > family ! = 5 | | this - > familyVersion ! = 1 ) {
qWarning ( ) < < " ParseEventsF5V1 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2019-12-01 23:21:18 +00:00
static const QMap < int , int > event_sizes = { { 1 , 2 } , { 8 , 4 } , { 9 , 3 } , { 0xa , 2 } , { 0xb , 5 } , { 0xc , 5 } , { 0xd , 0xd } } ;
2019-10-05 00:10:35 +00:00
if ( chunk_size < 1 ) {
// This does occasionally happen in F0V6.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
2019-10-08 14:44:18 +00:00
int elapsed , duration ;
2019-10-05 00:10:35 +00:00
do {
code = data [ pos + + ] ;
size = 3 ; // default size = 2 bytes time delta + 1 byte data
if ( event_sizes . contains ( code ) ) {
size = event_sizes [ code ] ;
}
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
2019-10-08 15:20:51 +00:00
if ( code ! = 0 ) { // Does this code really not have a timestamp? Never seen on F5V1, checked in F5V0.
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2019-10-05 00:10:35 +00:00
pos + = 2 ;
}
switch ( code ) {
2019-10-08 14:44:18 +00:00
//case 0x00: // never seen on F5V1
//case 0x01: // never seen on F5V1
case 0x02 : // Pressure adjustment
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos + + ] ) ) ;
break ;
2019-12-01 23:21:18 +00:00
case 0x03 : // Pressure Pulse
duration = data [ pos ] ; // TODO: is this a duration?
this - > AddEvent ( new PRS1PressurePulseEvent ( t , duration ) ) ;
break ;
2019-10-08 14:44:18 +00:00
case 0x04 : // Timed Breath
// TB events have a duration in 0.1s, based on the review of pressure waveforms.
// TODO: Ideally the starting time here would be adjusted here, but PRS1ParsedEvents
// currently assume integer seconds rather than ms, so that's done at import.
duration = data [ pos ] ;
this - > AddEvent ( new PRS1TimedBreathEvent ( t , duration ) ) ;
break ;
case 0x05 : // Obstructive Apnea
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x06 : // Clear Airway Apnea
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x07 : // Hypopnea
// TODO: How is this hypopnea different from event 0x8?
// NOTE: No additional (unknown) first byte as in F5V3 0x7, but see below.
// This seems closer to F5V3 0x0d or 0x0e.
elapsed = data [ pos ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x08 : // Hypopnea, note this is 0x7 in F5V3
// TODO: How is this hypopnea different from event 0x7?
// TODO: What is the first byte?
//data[pos+0]; // unknown first byte?
elapsed = data [ pos + 1 ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x09 : // Flow Limitation, note this is 0x8 in F5V3
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0a : // Vibratory Snore, note this is 0x9 in F5V3
// VS events are instantaneous flags with no duration, drawn on the official waveform.
// The current thinking is that these are the snores that cause a change in auto-titrating
// pressure. The snoring statistic above seems to be a total count. It's unclear whether
// the trigger for pressure change is severity or count or something else.
// no data bytes
this - > AddEvent ( new PRS1VibratorySnoreEvent ( t , 0 ) ) ;
break ;
case 0x0b : // Periodic Breathing, note this is 0xa in F5V3
// PB events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // confirmed to double in F5V0
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x0c : // Large Leak, note this is 0xb in F5V3
// LL events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // confirmed to double in F5V0
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x0d : // Statistics
// These appear every 2 minutes, so presumably summarize the preceding period.
this - > AddEvent ( new PRS1IPAPAverageEvent ( t , data [ pos + 0 ] ) ) ; // 00=IPAP
this - > AddEvent ( new PRS1IPAPLowEvent ( t , data [ pos + 1 ] ) ) ; // 01=IAP Low
this - > AddEvent ( new PRS1IPAPHighEvent ( t , data [ pos + 2 ] ) ) ; // 02=IAP High
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos + 3 ] ) ) ; // 03=Total leak (average?)
this - > AddEvent ( new PRS1RespiratoryRateEvent ( t , data [ pos + 4 ] ) ) ; // 04=Breaths Per Minute (average?)
this - > AddEvent ( new PRS1PatientTriggeredBreathsEvent ( t , data [ pos + 5 ] ) ) ; // 05=Patient Triggered Breaths (average?)
this - > AddEvent ( new PRS1MinuteVentilationEvent ( t , data [ pos + 6 ] ) ) ; // 06=Minute Ventilation (average?)
this - > AddEvent ( new PRS1TidalVolumeEvent ( t , data [ pos + 7 ] ) ) ; // 07=Tidal Volume (average?)
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 8 ] ) ) ; // 08=Snore count // TODO: not a VS on official waveform, but appears in flags and contributes to overall VS index
this - > AddEvent ( new PRS1EPAPAverageEvent ( t , data [ pos + 9 ] ) ) ; // 09=EPAP average
this - > AddEvent ( new PRS1LeakEvent ( t , data [ pos + 0xa ] ) ) ; // 0A=Leak (average?) new to F5V1 (originally found in F5V3)
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-10-08 14:44:18 +00:00
break ;
2019-10-05 00:10:35 +00:00
default :
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos ) ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
}
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing event bytes " ;
}
this - > duration = t ;
return ok ;
}
2019-10-22 14:24:40 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF5V2 = {
PRS1EPAPSetEvent : : TYPE ,
PRS1TimedBreathEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
//PRS1ClearAirwayEvent::TYPE, // not yet seen
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
//PRS1VibratorySnoreEvent::TYPE, // not yet seen
PRS1PeriodicBreathingEvent : : TYPE ,
//PRS1LargeLeakEvent::TYPE, // not yet seen
PRS1IPAPAverageEvent : : TYPE ,
PRS1IPAPLowEvent : : TYPE ,
PRS1IPAPHighEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1RespiratoryRateEvent : : TYPE ,
PRS1PatientTriggeredBreathsEvent : : TYPE ,
PRS1MinuteVentilationEvent : : TYPE ,
PRS1TidalVolumeEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
PRS1EPAPAverageEvent : : TYPE ,
PRS1LeakEvent : : TYPE ,
} ;
2019-10-05 00:10:35 +00:00
// 960T is F5V2
bool PRS1DataChunk : : ParseEventsF5V2 ( void )
{
if ( this - > family ! = 5 | | this - > familyVersion ! = 2 ) {
qWarning ( ) < < " ParseEventsF5V2 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
2019-10-04 23:51:34 +00:00
}
2019-10-05 00:10:35 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2019-10-08 20:42:47 +00:00
static const QMap < int , int > event_sizes = { { 0 , 4 } , { 1 , 2 } , { 3 , 4 } , { 8 , 3 } , { 9 , 4 } , { 0xa , 3 } , { 0xb , 5 } , { 0xc , 5 } , { 0xd , 5 } , { 0xe , 0xd } , { 0xf , 5 } , { 0x10 , 5 } , { 0x11 , 2 } , { 0x12 , 6 } } ;
2019-10-04 23:51:34 +00:00
if ( chunk_size < 1 ) {
// This does occasionally happen in F0V6.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
2019-05-24 21:08:51 +00:00
2019-10-08 20:42:47 +00:00
// F5V3 uses a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
static const float GAIN = 0.125 ; // TODO: this should be parameterized somewhere more logical
2019-10-04 23:51:34 +00:00
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
2019-05-24 21:08:51 +00:00
int t = 0 ;
2019-10-08 20:42:47 +00:00
int elapsed /*, duration, value*/ ;
2019-10-04 23:51:34 +00:00
do {
code = data [ pos + + ] ;
2014-04-17 05:58:57 +00:00
2019-10-04 23:51:34 +00:00
size = 3 ; // default size = 2 bytes time delta + 1 byte data
if ( event_sizes . contains ( code ) ) {
size = event_sizes [ code ] ;
}
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
if ( code ! = 0 & & code ! = 0x12 ) { // These two codes have no timestamp TODO: verify this applies to F5V012
t + = data [ pos ] /*| (data[pos+1] << 8)*/ ; // TODO: Is this really only 1 byte?
2019-10-08 14:44:18 +00:00
if ( data [ pos + 1 ] ! = 0 ) qWarning ( ) < < this - > sessionid < < " nonzero time? byte " < < hex ( startpos ) ;
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ;
2014-04-17 05:58:57 +00:00
pos + = 2 ;
2011-07-10 14:23:07 +00:00
}
2014-04-17 05:58:57 +00:00
2011-07-10 14:23:07 +00:00
switch ( code ) {
2019-10-08 20:42:47 +00:00
/*
2011-07-10 16:20:51 +00:00
case 0x00 : // Unknown (ASV Pressure value)
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2011-07-10 14:23:07 +00:00
// offset?
2019-10-04 23:51:34 +00:00
data0 = data [ pos + + ] ;
2014-04-17 05:58:57 +00:00
2019-10-04 23:51:34 +00:00
if ( ! data [ pos - 1 ] ) { // WTH???
data1 = data [ pos + + ] ;
2011-07-10 14:23:07 +00:00
}
2014-04-17 05:58:57 +00:00
2019-10-04 23:51:34 +00:00
if ( ! data [ pos - 1 ] ) {
//data2 = data[pos++];
2019-06-22 01:04:16 +00:00
pos + + ;
2011-07-10 14:23:07 +00:00
}
2014-04-17 05:58:57 +00:00
2011-07-10 14:23:07 +00:00
break ;
2014-04-17 05:58:57 +00:00
2011-07-27 09:21:53 +00:00
case 0x01 : // Unknown
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2019-05-24 21:08:51 +00:00
this - > AddEvent ( new PRS1UnknownValueEvent ( code , t , 0 , 0.1F ) ) ;
2011-07-27 09:21:53 +00:00
break ;
2019-10-08 20:42:47 +00:00
*/
case 0x02 : // Pressure adjustment
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos + + ] , GAIN ) ) ;
break ;
/*
2016-04-15 16:11:45 +00:00
case 0x03 : // BIPAP Pressure
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2016-04-15 16:11:45 +00:00
qDebug ( ) < < " 0x03 Observed in ASV data!!???? " ;
2019-10-04 23:51:34 +00:00
data0 = data [ pos + + ] ;
data1 = data [ pos + + ] ;
2016-04-15 16:11:45 +00:00
// data0/=10.0;
// data1/=10.0;
// session->AddEvent(new Event(t,CPAP_EAP, 0, data, 1));
// session->AddEvent(new Event(t,CPAP_IAP, 0, &data1, 1));
break ;
2019-10-08 20:42:47 +00:00
*/
case 0x04 : // Timed Breath
// TB events have a duration in 0.1s, based on the review of pressure waveforms.
// TODO: Ideally the starting time here would be adjusted here, but PRS1ParsedEvents
// currently assume integer seconds rather than ms, so that's done at import.
duration = data [ pos ] ;
this - > AddEvent ( new PRS1TimedBreathEvent ( t , duration ) ) ;
break ;
case 0x05 : // Obstructive Apnea
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
/*
2011-07-10 14:23:07 +00:00
case 0x06 :
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2012-01-12 05:57:36 +00:00
//code=CPAP_ClearAirway;
2019-10-04 23:51:34 +00:00
data0 = data [ pos + + ] ;
2019-05-24 21:08:51 +00:00
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - data0 , data0 ) ) ;
2011-07-27 09:21:53 +00:00
break ;
2019-10-08 20:42:47 +00:00
*/
2014-04-17 05:58:57 +00:00
2019-10-08 20:42:47 +00:00
case 0x07 : // Hypopnea
// NOTE: No additional (unknown) first byte as in F5V3 0x07, but see below.
// This seems closer to F5V3 0x0d or 0x0e.
// What's different about this an 0x08? This was seen in a PB at least once?
elapsed = data [ pos ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x08 : // Hypopnea, note this is 0x7 in F5V1
// TODO: How is this hypopnea different from event 0x9 and 0x7?
// NOTE: No additional (unknown) first byte as in F5V3 0x7, but see below.
// This seems closer to F5V3 0x0d or 0x0e.
elapsed = data [ pos ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
/*
2011-07-27 09:21:53 +00:00
case 0x09 : // ASV Codes
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
/ *
2019-05-24 21:08:51 +00:00
if ( this - > familyVersion < 2 ) {
2016-04-16 00:02:10 +00:00
//code=CPAP_FlowLimit;
2019-10-04 23:51:34 +00:00
data0 = data [ pos + + ] ;
2014-04-17 05:58:57 +00:00
2019-05-24 21:08:51 +00:00
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - data0 , data0 ) ) ;
2016-04-16 00:02:10 +00:00
} else {
2019-10-08 20:42:47 +00:00
* /
2019-10-04 23:51:34 +00:00
data0 = data [ pos + + ] ;
data1 = data [ pos + + ] ;
2011-07-27 09:21:53 +00:00
break ;
2019-10-08 20:42:47 +00:00
*/
case 0x0a : // Flow Limitation, note this is 0x9 in F5V1 and 0x8 in F5V3
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
break ;
/*
2011-07-10 14:23:07 +00:00
case 0x0b : // Cheyne Stokes
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2019-10-04 23:51:34 +00:00
data0 = ( ( unsigned char * ) data ) [ pos + 1 ] < < 8 | ( ( unsigned char * ) data ) [ pos ] ;
2016-03-04 21:27:33 +00:00
//data0*=2;
2014-04-17 05:58:57 +00:00
pos + = 2 ;
2019-10-04 23:51:34 +00:00
data1 = ( ( unsigned char * ) data ) [ pos ] ; //|data[pos+1] << 8
2014-04-17 05:58:57 +00:00
pos + = 1 ;
2011-07-10 14:23:07 +00:00
//tt-=delta;
2019-05-24 21:08:51 +00:00
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - data1 , data0 ) ) ;
2011-07-10 14:23:07 +00:00
break ;
2019-10-08 20:42:47 +00:00
*/
case 0x0c : // Periodic Breathing, note this is 0xb in F5V1 and 0xa in F5V3
// PB events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // confirmed to double in F5V0
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
/*
case 0x0d :
DUMP_EVENT ( ) ;
2012-01-12 05:57:36 +00:00
2019-10-04 23:51:34 +00:00
data0 = ( data [ pos + 1 ] < < 8 | data [ pos ] ) ;
2016-04-16 00:02:10 +00:00
data0 * = 2 ;
pos + = 2 ;
2019-10-04 23:51:34 +00:00
data1 = data [ pos + + ] ;
2019-05-24 21:08:51 +00:00
//tt = t - qint64(data1) * 1000L;
2011-07-10 14:23:07 +00:00
break ;
2019-10-08 20:42:47 +00:00
*/
case 0x0e : // Statistics, note this was 0x0d in F5V0 and F5V1
// These appear every 2 minutes, so presumably summarize the preceding period.
this - > AddEvent ( new PRS1IPAPAverageEvent ( t , data [ pos + 0 ] , GAIN ) ) ; // 00=IPAP
this - > AddEvent ( new PRS1IPAPLowEvent ( t , data [ pos + 1 ] , GAIN ) ) ; // 01=IAP Low
this - > AddEvent ( new PRS1IPAPHighEvent ( t , data [ pos + 2 ] , GAIN ) ) ; // 02=IAP High
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos + 3 ] ) ) ; // 03=Total leak (average?)
this - > AddEvent ( new PRS1RespiratoryRateEvent ( t , data [ pos + 4 ] ) ) ; // 04=Breaths Per Minute (average?)
this - > AddEvent ( new PRS1PatientTriggeredBreathsEvent ( t , data [ pos + 5 ] ) ) ; // 05=Patient Triggered Breaths (average?)
this - > AddEvent ( new PRS1MinuteVentilationEvent ( t , data [ pos + 6 ] ) ) ; // 06=Minute Ventilation (average?)
this - > AddEvent ( new PRS1TidalVolumeEvent ( t , data [ pos + 7 ] ) ) ; // 07=Tidal Volume (average?)
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 8 ] ) ) ; // 08=Snore count // TODO: not a VS on official waveform, but appears in flags and contributes to overall VS index
this - > AddEvent ( new PRS1EPAPAverageEvent ( t , data [ pos + 9 ] , GAIN ) ) ; // 09=EPAP average
this - > AddEvent ( new PRS1LeakEvent ( t , data [ pos + 0xa ] ) ) ; // 0A=Leak (average?) new to F5V1 (originally found in F5V3)
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-10-08 20:42:47 +00:00
break ;
/*
2016-04-15 16:11:45 +00:00
case 0x0f :
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2016-04-15 16:11:45 +00:00
qDebug ( ) < < " 0x0f Observed in ASV data!!???? " ;
2014-04-17 05:58:57 +00:00
2019-10-04 23:51:34 +00:00
data0 = data [ pos + 1 ] < < 8 | data [ pos ] ;
2016-04-15 16:11:45 +00:00
pos + = 2 ;
2019-10-04 23:51:34 +00:00
data1 = data [ pos ] ; //|data[pos+1] << 8
2016-04-15 16:11:45 +00:00
pos + = 1 ;
2019-05-24 21:08:51 +00:00
//tt -= qint64(data1) * 1000L;
2016-04-15 16:11:45 +00:00
//session->AddEvent(new Event(tt,cpapcode, 0, data, 2));
2011-07-10 14:23:07 +00:00
break ;
2011-12-10 12:14:48 +00:00
case 0x10 : // Unknown
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2019-10-04 23:51:34 +00:00
data0 = data [ pos + 1 ] < < 8 | data [ pos ] ;
2014-08-06 07:08:34 +00:00
pos + = 2 ;
2019-10-04 23:51:34 +00:00
data1 = data [ pos + + ] ;
2014-08-06 07:08:34 +00:00
2019-05-24 21:08:51 +00:00
this - > AddEvent ( new PRS1LargeLeakEvent ( t - data1 , data0 ) ) ;
2014-08-06 07:08:34 +00:00
// qDebug() << "0x10 Observed in ASV data!!????";
2019-10-04 23:51:34 +00:00
// data0 = data[pos++]; // << 8) | data[pos];
// data1 = data[pos++];
// data2 = data[pos++];
2011-12-10 12:14:48 +00:00
//session->AddEvent(new Event(t,cpapcode, 0, data, 3));
break ;
2016-04-15 16:11:45 +00:00
case 0x11 : // Not Leak Rate
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2016-04-15 16:11:45 +00:00
qDebug ( ) < < " 0x11 Observed in ASV data!!???? " ;
//if (!Code[24]) {
// Code[24]=new EventList(cpapcode,EVL_Event);
//}
2019-10-04 23:51:34 +00:00
//Code[24]->AddEvent(t,data[pos++]);
2011-12-10 12:14:48 +00:00
break ;
2014-04-17 05:58:57 +00:00
2016-04-15 16:11:45 +00:00
2011-12-10 12:14:48 +00:00
case 0x12 : // Summary
2019-10-08 20:42:47 +00:00
DUMP_EVENT ( ) ;
2011-12-10 12:14:48 +00:00
qDebug ( ) < < " 0x12 Observed in ASV data!!???? " ;
2019-10-04 23:51:34 +00:00
data0 = data [ pos + + ] ;
data1 = data [ pos + + ] ;
//data2 = data[pos + 1] << 8 | data[pos];
2014-04-17 05:58:57 +00:00
pos + = 2 ;
2011-12-10 12:14:48 +00:00
//session->AddEvent(new Event(t,cpapcode, 0, data,3));
break ;
2019-10-08 20:42:47 +00:00
*/
2019-10-04 23:51:34 +00:00
default :
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos ) ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
2011-12-10 12:14:48 +00:00
}
2019-10-04 23:51:34 +00:00
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
2014-04-17 05:58:57 +00:00
2019-10-04 23:51:34 +00:00
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing event bytes " ;
2011-12-10 12:14:48 +00:00
}
2013-10-25 10:39:30 +00:00
2019-10-04 23:51:34 +00:00
this - > duration = t ;
2011-12-10 12:14:48 +00:00
2019-10-04 23:51:34 +00:00
return ok ;
2011-12-10 12:14:48 +00:00
}
2019-07-25 02:42:00 +00:00
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 ) ;
// Generate the list of channels created by non-slice events for this machine.
// We can't just use the full list of non-slice events, since on some machines
// PS is generated by slice events (EPAP/IPAP average).
// TODO: convert supported to QSet and clean this up.
QSet < PRS1ParsedEventType > supportedNonSliceEvents = QSet < PRS1ParsedEventType > : : fromList ( QList < PRS1ParsedEventType > : : fromVector ( supported ) ) ;
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 ) ;
}
}
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 ) ;
}
}
// 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 ;
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 ;
}
}
2020-01-07 02:45:52 +00:00
if ( updated ) {
// Write out any pending end-of-slice events.
FinishSlice ( ) ;
}
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 ) ;
}
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 ;
}
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-10-22 18:48:34 +00:00
// Unintentional leak calculation, see zMaskProfile:calcLeak in calcs.cpp for explanation
2019-11-07 19:19:58 +00:00
m_calcLeaks = p_profile - > cpap - > calculateUnintentionalLeaks ( ) ;
if ( m_calcLeaks ) {
2019-10-22 18:48:34 +00:00
// Only needed for machines that don't support it directly.
2019-11-07 19:19:58 +00:00
m_calcLeaks = ( supported . contains ( PRS1LeakEvent : : TYPE ) = = false ) ;
2019-10-22 18:48:34 +00:00
}
2019-11-07 19:19:58 +00:00
m_lpm4 = p_profile - > cpap - > custom4cmH2OLeaks ( ) ;
2019-10-22 18:48:34 +00:00
EventDataType lpm20 = p_profile - > cpap - > custom20cmH2OLeaks ( ) ;
2019-11-07 19:19:58 +00:00
EventDataType lpm = lpm20 - m_lpm4 ;
m_ppm = lpm / 16.0 ;
2019-10-22 18:48:34 +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 ;
}
// Skip zero-length PB or LL (or unknown duration 0E) events
if ( ( e - > m_type = = PRS1PeriodicBreathingEvent : : TYPE | | e - > m_type = = PRS1LargeLeakEvent : : TYPE | | e - > m_type = = PRS1UnknownDurationEvent : : TYPE ) & &
( 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.
//
// It has also happened once in a similar scenario for PB and 0E, even when
// 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.
if ( event - > family = = 3 & & event - > familyVersion = = 3 ) {
// In F3V3, each slice has its own chunk, so the initial call to UpdateCurrentSlice()
// 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 {
// For all other machines, the event's time stamp will be within bounds of its slice, so
// 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 ) ) {
// LL and PRS1_0E at the beginning of a mask-on session sometimes start 1 second early,
// so suppress that warning.
if ( ( * m_currentSlice ) . start - t > 1000 | | ( e - > m_type ! = PRS1LargeLeakEvent : : TYPE & & e - > m_type ! = PRS1UnknownDurationEvent : : TYPE ) ) {
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 ;
}
// TODO: This needs to be special-cased for F3V3 due to its weird interval-based event format
// 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.
if ( ! ( event - > family = = 3 & & event - > familyVersion = = 3 ) ) {
// 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 ( ) ) {
// This has only been seen once, with corrupted data, in which the summary and event
// files each contained multiple conflicting sessions (all brief) with the same ID.
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.
//
// These actually get reported by the machines, but they cause "unordered time" warnings
// 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 ] ;
ChannelID channel = NoChannel , PS , VS2 , LEAK ;
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
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 :
case PRS1UnknownDurationEvent : : TYPE :
// 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 PRS1TotalLeakEvent : : TYPE :
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ;
// F0 up through F0V6 doesn't appear to report unintentional leak.
// TODO: decide whether to keep this here, shouldn't keep it here just because it's "quicker".
// TODO: compare this value for the reported value for F5V1 and higher?
// TODO: Fix this for 0.125 gain: it assumes 0.1 (dividing by 10.0)...
// Or omit, because machines with 0.125 gain report unintentional leak directly.
if ( m_calcLeaks ) { // Much Quicker doing this here than the recalc method.
EventDataType leak = e - > m_value ;
leak - = ( ( ( m_currentPressure / 10.0f ) - 4.0 ) * m_ppm + m_lpm4 ) ;
if ( leak < 0 ) leak = 0 ;
LEAK = * channels . at ( 1 ) ;
AddEvent ( LEAK , t , leak , 1 ) ;
}
break ;
2019-10-22 18:48:34 +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.
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ; // Snore count
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.
VS2 = * channels . at ( 1 ) ;
AddEvent ( VS2 , t , 0 , 1 ) ;
}
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-10-19 21:35:45 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF3V6 = {
PRS1TimedBreathEvent : : TYPE ,
PRS1IPAPAverageEvent : : TYPE ,
PRS1EPAPAverageEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1RespiratoryRateEvent : : TYPE ,
PRS1PatientTriggeredBreathsEvent : : TYPE ,
PRS1MinuteVentilationEvent : : TYPE ,
PRS1TidalVolumeEvent : : TYPE ,
PRS1Test2Event : : TYPE ,
PRS1Test1Event : : TYPE ,
PRS1SnoreEvent : : TYPE , // No individual VS, only snore count
PRS1LeakEvent : : TYPE ,
PRS1PressurePulseEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1RERAEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
PRS1ApneaAlarmEvent : : TYPE ,
// No FL?
} ;
2019-05-24 23:41:42 +00:00
// 1030X, 11030X series
2019-07-26 01:44:36 +00:00
// based on ParseEventsF5V3, updated for F3V6
2019-05-24 23:41:42 +00:00
bool PRS1DataChunk : : ParseEventsF3V6 ( void )
{
2019-07-26 01:44:36 +00:00
if ( this - > family ! = 3 | | this - > familyVersion ! = 6 ) {
qWarning ( ) < < " ParseEventsF3V6 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2019-07-27 02:29:23 +00:00
static const int minimum_sizes [ ] = { 2 , 3 , 0xe , 3 , 3 , 3 , 4 , 5 , 3 , 5 , 3 , 3 , 2 , 2 , 2 , 2 } ;
2019-07-26 01:44:36 +00:00
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
2019-05-27 16:38:55 +00:00
2019-07-26 01:44:36 +00:00
if ( chunk_size < 1 ) {
// This does occasionally happen.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
// F3V6 uses a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
static const float GAIN = 0.125 ; // TODO: this should be parameterized somewhere more logical
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
2019-07-27 02:13:26 +00:00
int elapsed , duration ;
2019-07-26 01:44:36 +00:00
do {
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for event " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
if ( size < minimum_sizes [ code ] ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " too small " < < size < < " < " < < minimum_sizes [ code ] ;
ok = false ;
break ;
}
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
pos + = 2 ;
switch ( code ) {
2019-07-27 02:29:23 +00:00
// case 0x00?
2019-07-26 01:44:36 +00:00
case 1 : // Timed Breath
// TB events have a duration in 0.1s, based on the review of pressure waveforms.
// TODO: Ideally the starting time here would be adjusted here, but PRS1ParsedEvents
// currently assume integer seconds rather than ms, so that's done at import.
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ;
2019-07-26 01:44:36 +00:00
// TODO: make sure F3 import logic matches F5 in adjusting TB start time
this - > AddEvent ( new PRS1TimedBreathEvent ( t , duration ) ) ;
break ;
case 2 : // Statistics
// These appear every 2 minutes, so presumably summarize the preceding period.
2019-08-20 15:38:55 +00:00
//data[pos+0]; // TODO: 0 = ???
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1IPAPAverageEvent ( t , data [ pos + 2 ] , GAIN ) ) ; // 02=IPAP
2019-10-11 00:13:39 +00:00
this - > AddEvent ( new PRS1EPAPAverageEvent ( t , data [ pos + 1 ] , GAIN ) ) ; // 01=EPAP, needs to be added second to calculate PS
2019-08-20 15:38:55 +00:00
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos + 3 ] ) ) ; // 03=Total leak (average?)
this - > AddEvent ( new PRS1RespiratoryRateEvent ( t , data [ pos + 4 ] ) ) ; // 04=Breaths Per Minute (average?)
this - > AddEvent ( new PRS1PatientTriggeredBreathsEvent ( t , data [ pos + 5 ] ) ) ; // 05=Patient Triggered Breaths (average?)
this - > AddEvent ( new PRS1MinuteVentilationEvent ( t , data [ pos + 6 ] ) ) ; // 06=Minute Ventilation (average?)
this - > AddEvent ( new PRS1TidalVolumeEvent ( t , data [ pos + 7 ] ) ) ; // 07=Tidal Volume (average?)
this - > AddEvent ( new PRS1Test2Event ( t , data [ pos + 8 ] ) ) ; // 08=Flow???
this - > AddEvent ( new PRS1Test1Event ( t , data [ pos + 9 ] ) ) ; // 09=TMV???
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 0xa ] ) ) ; // 0A=Snore count // TODO: not a VS on official waveform, but appears in flags and contributes to overall VS index
this - > AddEvent ( new PRS1LeakEvent ( t , data [ pos + 0xb ] ) ) ; // 0B=Leak (average?)
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-07-26 01:44:36 +00:00
break ;
2019-07-27 02:13:26 +00:00
case 0x03 : // Pressure Pulse
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ; // TODO: is this a duration?
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1PressurePulseEvent ( t , duration ) ) ;
break ;
2019-07-27 02:13:26 +00:00
case 0x04 : // Obstructive Apnea
2019-07-26 01:44:36 +00:00
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
2019-07-27 02:13:26 +00:00
case 0x05 : // Clear Airway Apnea
2019-07-26 01:44:36 +00:00
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
2019-07-27 02:13:26 +00:00
case 0x06 : // Hypopnea
2019-07-26 01:44:36 +00:00
// TODO: How is this hypopnea different from events 0xd and 0xe?
// TODO: What is the first byte?
2019-08-20 15:38:55 +00:00
//data[pos+0]; // unknown first byte?
elapsed = data [ pos + 1 ] ; // based on sample waveform, the hypopnea is over after this
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
2019-07-27 02:13:26 +00:00
case 0x07 : // Periodic Breathing
2019-07-26 01:44:36 +00:00
// PB events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ;
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
2019-07-27 02:13:26 +00:00
case 0x08 : // RERA
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ; // based on sample waveform, the RERA is over after this
2019-07-27 02:13:26 +00:00
this - > AddEvent ( new PRS1RERAEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x09 : // Large Leak
2019-07-26 01:44:36 +00:00
// LL events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ;
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
break ;
2019-07-27 02:13:26 +00:00
case 0x0a : // Hypopnea
2019-07-26 01:44:36 +00:00
// TODO: Why does this hypopnea have a different event code?
// fall through
case 0x0b : // Hypopnea
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating hypopneas ourselves. Their official definition
// is 40% reduction in flow lasting at least 10s.
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ;
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1HypopneaEvent ( t - duration , 0 ) ) ;
break ;
2019-09-19 14:33:23 +00:00
case 0x0c : // Apnea Alarm
// no additional data
2019-09-19 20:43:24 +00:00
this - > AddEvent ( new PRS1ApneaAlarmEvent ( t , 0 ) ) ;
2019-09-19 14:33:23 +00:00
break ;
2019-07-27 02:29:23 +00:00
// case 0x0d?
// case 0x0e?
// case 0x0f?
2019-07-26 01:44:36 +00:00
default :
2020-03-23 00:00:09 +00:00
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
2019-07-26 01:44:36 +00:00
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos - 1 , size + 1 ) ) ;
break ;
}
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
this - > duration = t ;
return ok ;
}
2019-10-19 21:35:45 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF3V3 = {
PRS1IPAPAverageEvent : : TYPE ,
PRS1EPAPAverageEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1TidalVolumeEvent : : TYPE ,
PRS1FlowRateEvent : : TYPE ,
PRS1PatientTriggeredBreathsEvent : : TYPE ,
PRS1RespiratoryRateEvent : : TYPE ,
PRS1MinuteVentilationEvent : : TYPE ,
PRS1LeakEvent : : TYPE ,
2019-11-12 22:35:13 +00:00
PRS1HypopneaCount : : TYPE ,
PRS1ClearAirwayCount : : TYPE ,
PRS1ObstructiveApneaCount : : TYPE ,
2019-10-19 21:35:45 +00:00
// No PP, FL, VS, RERA, PB, LL
// No TB
} ;
2019-07-26 01:44:36 +00:00
// 1061T, 1160P series
2019-05-25 00:09:53 +00:00
bool PRS1DataChunk : : ParseEventsF3V3 ( void )
{
2019-07-26 01:44:36 +00:00
// NOTE: Older ventilators (BiPAP S/T and AVAPS) machines don't use timestamped events like everything else.
// Instead, they use a fixed interval format like waveforms do (see PRS1_HTYPE_INTERVAL).
2019-05-25 00:09:53 +00:00
if ( this - > family ! = 3 | | this - > familyVersion ! = 3 ) {
qWarning ( ) < < " ParseEventsF3V3 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
2019-06-04 02:01:02 +00:00
return false ;
2019-05-25 00:09:53 +00:00
}
2019-10-21 01:58:36 +00:00
if ( this - > fileVersion = = 3 ) {
// NOTE: The original comment in the header for ParseF3EventsV3 said there was a 1060P with fileVersion 3.
// We've never seen that, so warn if it ever shows up.
qWarning ( ) < < " F3V3 event file with fileVersion 3? " ;
}
2019-05-25 00:09:53 +00:00
2019-11-12 22:35:13 +00:00
int t = 0 ;
2019-10-09 14:24:29 +00:00
static const int record_size = 0x10 ;
int size = this - > m_data . size ( ) / record_size ;
CHECK_VALUE ( this - > m_data . size ( ) % record_size , 0 ) ;
2019-05-25 00:09:53 +00:00
unsigned char * h = ( unsigned char * ) this - > m_data . data ( ) ;
2016-02-28 15:11:53 +00:00
2019-11-12 22:35:13 +00:00
static const qint64 block_duration = 120 ;
2016-02-28 15:11:53 +00:00
2019-10-09 14:24:29 +00:00
// Make sure the assumptions here agree with the header
CHECK_VALUE ( this - > htype , PRS1_HTYPE_INTERVAL ) ;
CHECK_VALUE ( this - > interval_count , size ) ;
2019-11-12 22:35:13 +00:00
CHECK_VALUE ( this - > interval_seconds , block_duration ) ;
2019-10-09 14:24:29 +00:00
for ( auto & channel : this - > waveformInfo ) {
CHECK_VALUE ( channel . interleave , 1 ) ;
}
2019-05-18 23:17:55 +00:00
2016-02-28 15:11:53 +00:00
for ( int x = 0 ; x < size ; x + + ) {
2019-11-12 22:35:13 +00:00
// Use the timestamp of the end of this interval, to be consistent with other parsers,
// but see note below regarding the duration of the final interval.
t + = block_duration ;
2019-10-19 21:35:45 +00:00
// TODO: The duration of the final interval isn't clearly defined in this format:
// there appears to be no way (apart from looking at the summary or waveform data)
// to determine the end time, which may truncate the last interval.
//
// TODO: What if there are multiple "final" intervals in a session due to multiple
// mask-on slices?
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1IPAPAverageEvent ( t , h [ 0 ] | ( h [ 1 ] < < 8 ) ) ) ;
this - > AddEvent ( new PRS1EPAPAverageEvent ( t , h [ 2 ] | ( h [ 3 ] < < 8 ) ) ) ;
2019-05-25 00:09:53 +00:00
this - > AddEvent ( new PRS1TotalLeakEvent ( t , h [ 4 ] ) ) ;
this - > AddEvent ( new PRS1TidalVolumeEvent ( t , h [ 5 ] ) ) ;
this - > AddEvent ( new PRS1FlowRateEvent ( t , h [ 6 ] ) ) ;
this - > AddEvent ( new PRS1PatientTriggeredBreathsEvent ( t , h [ 7 ] ) ) ;
this - > AddEvent ( new PRS1RespiratoryRateEvent ( t , h [ 8 ] ) ) ;
2019-10-09 14:24:29 +00:00
if ( h [ 9 ] < 13 | | h [ 9 ] > 84 ) UNEXPECTED_VALUE ( h [ 9 ] , " 13-84 " ) ; // not sure what this is.. encore doesn't graph it.
CHECK_VALUES ( h [ 10 ] , 0 , 8 ) ; // 8 shows as a Low Pressure (LP) alarm
2019-05-25 00:09:53 +00:00
this - > AddEvent ( new PRS1MinuteVentilationEvent ( t , h [ 11 ] ) ) ;
2019-11-12 22:35:13 +00:00
this - > AddEvent ( new PRS1HypopneaCount ( t , h [ 12 ] ) ) ; // count of hypopnea events
this - > AddEvent ( new PRS1ClearAirwayCount ( t , h [ 13 ] ) ) ; // count of clear airway events
this - > AddEvent ( new PRS1ObstructiveApneaCount ( t , h [ 14 ] ) ) ; // count of obstructive events
2019-10-09 14:24:29 +00:00
this - > AddEvent ( new PRS1LeakEvent ( t , h [ 15 ] ) ) ;
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2016-02-28 15:11:53 +00:00
2019-10-09 14:24:29 +00:00
h + = record_size ;
2016-02-28 15:11:53 +00:00
}
2019-10-09 14:24:29 +00:00
this - > duration = t ;
2016-03-03 23:26:20 +00:00
return true ;
2016-02-28 15:11:53 +00:00
}
2019-05-25 00:09:53 +00:00
2019-05-27 16:38:55 +00:00
#if 0
// Currently unused, apparently an abandoned effort to massage F0 pressure/IPAP/EPAP data.
2018-05-05 07:14:44 +00:00
extern EventDataType CatmullRomSpline ( EventDataType p0 , EventDataType p1 , EventDataType p2 , EventDataType p3 , EventDataType t = 0.5 ) ;
2016-03-07 13:54:14 +00:00
void SmoothEventList ( Session * session , EventList * ev , ChannelID code )
{
if ( ! ev ) return ;
int cnt = ev - > count ( ) ;
if ( cnt > 4 ) {
EventList * smooth = new EventList ( EVL_Event , ev - > gain ( ) ) ;
smooth - > setFirst ( ev - > first ( ) ) ;
smooth - > AddEvent ( ev - > time ( 0 ) , ev - > raw ( 0 ) ) ;
2018-05-05 07:14:44 +00:00
EventDataType p0 , p1 , p2 , p3 , v ;
2016-03-07 13:54:14 +00:00
for ( int i = 1 ; i < cnt - 2 ; + + i ) {
qint64 time = ev - > time ( i ) ;
qint64 time2 = ev - > time ( i + 1 ) ;
qint64 diff = time2 - time ;
// these aren't evenly spaced... spline won't work here.
p0 = ev - > raw ( i - 1 ) ;
p1 = ev - > raw ( i ) ;
p2 = ev - > raw ( i + 1 ) ;
p3 = ev - > raw ( i + 2 ) ;
smooth - > AddEvent ( time , p1 ) ;
// int df = p2-p1;
// if (df > 0) {
// qint64 inter = diff/(df+1);
// qint64 t = time+inter;
// for (int j=0; j<df; ++j) {
// smooth->AddEvent(t, p1+j);
// t+=inter;
// }
// } else if (df<0) {
// df = abs(df);
// qint64 inter = diff/(df+1);
// qint64 t = time+inter;
// for (int j=0; j<df; ++j) {
// smooth->AddEvent(t, p1-j);
// t+=inter;
// }
// }
// don't want to use Catmull here...
v = CatmullRomSpline ( p0 , p1 , p2 , p3 , 0.25 ) ;
smooth - > AddEvent ( time + diff * 0.25 , v ) ;
v = CatmullRomSpline ( p0 , p1 , p2 , p3 , 0.5 ) ;
smooth - > AddEvent ( time + diff * 0.5 , v ) ;
v = CatmullRomSpline ( p0 , p1 , p2 , p3 , 0.75 ) ;
smooth - > AddEvent ( time + diff * 0.75 , v ) ;
}
smooth - > AddEvent ( ev - > time ( cnt - 2 ) , ev - > raw ( cnt - 2 ) ) ;
smooth - > AddEvent ( ev - > time ( cnt - 1 ) , ev - > raw ( cnt - 1 ) ) ;
session - > eventlist [ code ] . removeAll ( ev ) ;
delete ev ;
session - > eventlist [ code ] . append ( smooth ) ;
}
}
2019-05-27 16:38:55 +00:00
# endif
2016-03-07 13:54:14 +00:00
2019-05-25 21:00:44 +00:00
2019-10-22 16:30:42 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF0V23 = {
PRS1PressureSetEvent : : TYPE ,
PRS1IPAPSetEvent : : TYPE ,
PRS1EPAPSetEvent : : TYPE ,
PRS1PressurePulseEvent : : TYPE ,
PRS1RERAEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
PRS1VibratorySnoreEvent : : TYPE ,
2019-10-30 22:38:13 +00:00
PRS1UnknownDurationEvent : : TYPE ,
2019-10-22 16:30:42 +00:00
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
PRS1SnoresAtPressureEvent : : TYPE ,
} ;
2019-10-24 01:10:56 +00:00
// 750P is F0V2; 550P is F0V2/F0V3 (properties.txt sometimes says F0V3, data files always say F0V2); 450P is F0V3
2019-10-09 17:35:02 +00:00
bool PRS1DataChunk : : ParseEventsF0V23 ( )
2019-05-25 21:00:44 +00:00
{
2019-10-03 20:50:59 +00:00
if ( this - > family ! = 0 | | this - > familyVersion < 2 | | this - > familyVersion > 3 ) {
qWarning ( ) < < " ParseEventsF0V23 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
2019-10-04 16:08:16 +00:00
// All sample machines with FamilyVersion 3 in the properties.txt file have familyVersion 2 in their .001/.002/.005 files!
// We should flag an actual familyVersion 3 file if we ever encounter one!
CHECK_VALUE ( this - > familyVersion , 2 ) ;
2019-10-03 23:42:28 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
static const QMap < int , int > event_sizes = { { 1 , 2 } , { 3 , 4 } , { 0xb , 4 } , { 0xd , 2 } , { 0xe , 5 } , { 0xf , 5 } , { 0x10 , 5 } , { 0x11 , 4 } , { 0x12 , 4 } } ;
if ( chunk_size < 1 ) {
// This does occasionally happen in F0V6.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
2019-10-03 20:50:59 +00:00
2019-10-03 23:42:28 +00:00
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
2019-10-04 01:18:52 +00:00
int elapsed , duration , value ;
2019-10-03 23:42:28 +00:00
do {
code = data [ pos + + ] ;
2019-10-03 20:50:59 +00:00
2019-10-03 23:42:28 +00:00
size = 3 ; // default size = 2 bytes time delta + 1 byte data
if ( event_sizes . contains ( code ) ) {
size = event_sizes [ code ] ;
2019-10-03 20:50:59 +00:00
}
2019-10-03 23:42:28 +00:00
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
if ( code ! = 0x12 ) { // This one event has no timestamp in F0V6
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2019-10-03 20:50:59 +00:00
pos + = 2 ;
}
switch ( code ) {
2019-10-04 16:08:16 +00:00
case 0x00 : // ??? So far only seen on 451P and 551P occasionally, usually no more than once per session
// A nonzero delta corresponds to an N-second gap in data (value was 0x85, only seen once). Look for more.
2020-01-12 22:27:12 +00:00
if ( sessionid ! = 122 ) CHECK_VALUE ( data [ startpos ] , 0 ) ; // skip the onc occurrence already seen
2019-10-04 16:08:16 +00:00
CHECK_VALUE ( data [ startpos + 1 ] , 0 ) ;
if ( data [ pos ] < 0x80 | | data [ pos ] > 0x85 ) {
UNEXPECTED_VALUE ( data [ pos ] , " 0x80-0x85 " ) ;
DUMP_EVENT ( ) ;
}
if ( this - > familyVersion = = 3 ) DUMP_EVENT ( ) ;
break ;
2019-10-04 01:18:52 +00:00
//case 0x01: // never seen
case 0x02 : // Pressure adjustment
// See notes in ParseEventsF0V6.
this - > AddEvent ( new PRS1PressureSetEvent ( t , data [ pos ] ) ) ;
break ;
case 0x03 : // Pressure adjustment (bi-level)
// See notes in ParseEventsF0V6.
this - > AddEvent ( new PRS1IPAPSetEvent ( t , data [ pos + 1 ] ) ) ;
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos ] ) ) ; // EPAP needs to be added second to calculate PS
break ;
case 0x04 : // Pressure Pulse
duration = data [ pos ] ; // TODO: is this a duration?
this - > AddEvent ( new PRS1PressurePulseEvent ( t , duration ) ) ;
break ;
case 0x05 : // RERA
elapsed = data [ pos + + ] ;
this - > AddEvent ( new PRS1RERAEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x06 : // Obstructive Apnea
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x07 : // Clear Airway Apnea
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
//case 0x08: // never seen
//case 0x09: // never seen
case 0x0a : // Hypopnea
// TODO: How is this hypopnea different from events 0xb, [0x14 and 0x15 on F0V6]?
elapsed = data [ pos + + ] ;
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0b : // Hypopnea
// TODO: How is this hypopnea different from events 0xa, [0x14 and 0x15 on F0V6]?
// TODO: What is the first byte?
//data[pos+0]; // unknown first byte?
elapsed = data [ pos + 1 ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0c : // Flow Limitation
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0d : // Vibratory Snore
// VS events are instantaneous flags with no duration, drawn on the official waveform.
// The current thinking is that these are the snores that cause a change in auto-titrating
// pressure. The snoring statistics below seem to be a total count. It's unclear whether
// the trigger for pressure change is severity or count or something else.
// no data bytes
this - > AddEvent ( new PRS1VibratorySnoreEvent ( t , 0 ) ) ;
break ;
case 0x0e : // ???
// 5 bytes like PB and LL, but what is it?
// TODO: does duration double like F0V4?
duration = ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // this looks like a 16-bit value, so may be duration like PB?
elapsed = data [ pos + 2 ] ; // this is always 60 seconds unless it's at the end, so it seems like elapsed
CHECK_VALUES ( elapsed , 60 , 0 ) ;
this - > AddEvent ( new PRS1UnknownDurationEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x0f : // Periodic Breathing
// PB events are reported some time after they conclude, and they do have a reported duration.
// NOTE: F0V2 does NOT double this like F0V6 does
if ( this - > familyVersion = = 3 ) // double-check whether there's doubling on F0V3
DUMP_EVENT ( ) ;
duration = ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x10 : // Large Leak
// LL events are reported some time after they conclude, and they do have a reported duration.
2019-10-04 22:53:10 +00:00
// NOTE: F0V2 does NOT double this like F0V4 and F0V6 does
2019-10-04 01:18:52 +00:00
if ( this - > familyVersion = = 3 ) // double-check whether there's doubling on F0V3
DUMP_EVENT ( ) ;
duration = ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
break ;
2019-10-04 22:53:10 +00:00
case 0x11 : // Statistics
2019-10-04 01:18:52 +00:00
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos ] ) ) ;
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 1 ] ) ) ;
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-10-04 01:18:52 +00:00
break ;
case 0x12 : // Snore count per pressure
// Some sessions (with lots of ramps) have multiple of these, each with a
// different pressure. The total snore count across all of them matches the
// total found in the stats event.
if ( data [ pos ] ! = 0 ) {
CHECK_VALUES ( data [ pos ] , 1 , 2 ) ; // 0 = CPAP pressure, 1 = bi-level EPAP, 2 = bi-level IPAP
}
//CHECK_VALUE(data[pos+1], 0x78); // pressure
//CHECK_VALUE(data[pos+2], 1); // 16-bit snore count
//CHECK_VALUE(data[pos+3], 0);
value = ( data [ pos + 2 ] | ( data [ pos + 3 ] < < 8 ) ) ;
this - > AddEvent ( new PRS1SnoresAtPressureEvent ( t , data [ pos ] , data [ pos + 1 ] , value ) ) ;
break ;
2019-10-04 22:53:10 +00:00
default :
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos ) ) ;
2019-11-09 20:09:02 +00:00
ok = false ; // unlike F0V6, we don't know the size of unknown events, so we can't recover
2019-10-04 22:53:10 +00:00
break ;
2019-10-03 20:50:59 +00:00
}
2019-10-03 23:42:28 +00:00
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing event bytes " ;
2019-10-03 20:50:59 +00:00
}
2019-10-03 23:42:28 +00:00
2019-10-03 20:50:59 +00:00
this - > duration = t ;
2019-10-03 23:42:28 +00:00
return ok ;
2019-10-03 20:50:59 +00:00
}
2019-10-22 16:30:42 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF0V4 = {
PRS1PressureSetEvent : : TYPE ,
PRS1IPAPSetEvent : : TYPE ,
PRS1EPAPSetEvent : : TYPE ,
PRS1AutoPressureSetEvent : : TYPE ,
PRS1PressurePulseEvent : : TYPE ,
PRS1RERAEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
PRS1VibratorySnoreEvent : : TYPE ,
2019-10-30 22:38:13 +00:00
PRS1UnknownDurationEvent : : TYPE ,
2019-10-22 16:30:42 +00:00
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
2019-10-29 18:06:57 +00:00
PRS1PressureAverageEvent : : TYPE ,
2020-01-28 16:26:31 +00:00
PRS1FlexPressureAverageEvent : : TYPE ,
2019-10-22 16:30:42 +00:00
PRS1SnoresAtPressureEvent : : TYPE ,
} ;
2019-10-24 01:10:56 +00:00
// 460P, 560P[BT], 660P, 760P are F0V4
2019-10-09 17:35:02 +00:00
bool PRS1DataChunk : : ParseEventsF0V4 ( )
2019-10-03 20:50:59 +00:00
{
if ( this - > family ! = 0 | | this - > familyVersion ! = 4 ) {
qWarning ( ) < < " ParseEventsF0V4 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
2019-08-14 01:09:55 +00:00
return false ;
}
2019-10-04 19:10:08 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2019-10-04 22:53:10 +00:00
static const QMap < int , int > event_sizes = { { 0 , 4 } , { 2 , 4 } , { 3 , 3 } , { 0xb , 4 } , { 0xd , 2 } , { 0xe , 5 } , { 0xf , 5 } , { 0x10 , 5 } , { 0x11 , 5 } , { 0x12 , 4 } } ;
2019-10-04 19:10:08 +00:00
if ( chunk_size < 1 ) {
// This does occasionally happen in F0V6.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
2019-05-25 21:00:44 +00:00
2019-10-04 19:10:08 +00:00
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
2019-10-04 22:53:10 +00:00
int elapsed , duration , value ;
2020-01-28 16:26:31 +00:00
bool is_bilevel = false ;
2019-10-04 19:10:08 +00:00
do {
2019-10-03 23:42:28 +00:00
code = data [ pos + + ] ;
2014-04-17 05:58:57 +00:00
2019-10-04 19:10:08 +00:00
size = 3 ; // default size = 2 bytes time delta + 1 byte data
if ( event_sizes . contains ( code ) ) {
size = event_sizes [ code ] ;
2011-12-10 12:14:48 +00:00
}
2019-10-04 19:10:08 +00:00
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
if ( code ! = 0x12 ) { // This one event has no timestamp in F0V6
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2014-04-17 05:58:57 +00:00
pos + = 2 ;
2011-12-10 12:14:48 +00:00
}
2014-04-17 05:58:57 +00:00
2011-12-10 12:14:48 +00:00
switch ( code ) {
2019-10-04 22:53:10 +00:00
//case 0x00: // never seen
// NOTE: the original code thought 0x00 had 2 data bytes, unlike the 1 in F0V23.
// We don't have any sample data with this event, so it's left out here.
case 0x01 : // Pressure adjustment: note this was 0x02 in F0V23 and is 0x01 in F0V6
// See notes in ParseEventsF0V6.
this - > AddEvent ( new PRS1PressureSetEvent ( t , data [ pos ] ) ) ;
break ;
case 0x02 : // Pressure adjustment (bi-level): note that this was 0x03 in F0V23 and is 0x02 in F0V6
// See notes above on interpolation.
this - > AddEvent ( new PRS1IPAPSetEvent ( t , data [ pos + 1 ] ) ) ;
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos ] ) ) ; // EPAP needs to be added second to calculate PS
2020-01-28 16:26:31 +00:00
is_bilevel = true ;
2019-10-04 22:53:10 +00:00
break ;
case 0x03 : // Adjust Opti-Start pressure
// On F0V4 this occasionally shows up in the middle of a session.
// In that cases, the new pressure corresponds to the next night's Opti-Start
// pressure. It does not appear to have any effect on the current night's pressure,
// though presumaby it could if there's a long gap between sessions.
// See F0V6 event 3 for comparison.
// TODO: Does this occur in bi-level mode?
this - > AddEvent ( new PRS1AutoPressureSetEvent ( t , data [ pos ] ) ) ;
break ;
case 0x04 : // Pressure Pulse
duration = data [ pos ] ; // TODO: is this a duration?
this - > AddEvent ( new PRS1PressurePulseEvent ( t , duration ) ) ;
break ;
case 0x05 : // RERA
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1RERAEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x06 : // Obstructive Apnea
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x07 : // Clear Airway Apnea
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
//case 0x08: // never seen
//case 0x09: // never seen
case 0x0a : // Hypopnea
// TODO: How is this hypopnea different from events 0xb, [0x14 and 0x15 on F0V6]?
elapsed = data [ pos + + ] ;
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0b : // Hypopnea
// TODO: How is this hypopnea different from events 0xa, 0x14 and 0x15?
// TODO: What is the first byte?
//data[pos+0]; // unknown first byte?
elapsed = data [ pos + 1 ] ; // based on sample waveform, the hypopnea is over after this
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0c : // Flow Limitation
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
elapsed = data [ pos ] ;
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x0d : // Vibratory Snore
// VS events are instantaneous flags with no duration, drawn on the official waveform.
// The current thinking is that these are the snores that cause a change in auto-titrating
// pressure. The snoring statistics below seem to be a total count. It's unclear whether
// the trigger for pressure change is severity or count or something else.
// no data bytes
this - > AddEvent ( new PRS1VibratorySnoreEvent ( t , 0 ) ) ;
break ;
case 0x0e : // ???
// 5 bytes like PB and LL, but what is it?
// TODO: does duration double like it does for PB/LL?
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // this looks like a 16-bit value, so may be duration like PB?
elapsed = data [ pos + 2 ] ; // this is always 60 seconds unless it's at the end, so it seems like elapsed
CHECK_VALUES ( elapsed , 60 , 0 ) ;
this - > AddEvent ( new PRS1UnknownDurationEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x0f : // Periodic Breathing
// PB events are reported some time after they conclude, and they do have a reported duration.
// NOTE: This (and F0V6) doubles the duration, unlike F0V23.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x10 : // Large Leak
// LL events are reported some time after they conclude, and they do have a reported duration.
// NOTE: This (and F0V6) doubles the duration, unlike F0V23.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
elapsed = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
break ;
case 0x11 : // Statistics
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos ] ) ) ;
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 1 ] ) ) ;
2016-01-22 11:58:09 +00:00
2020-01-28 16:26:31 +00:00
value = data [ pos + 2 ] ;
if ( is_bilevel ) {
// For bi-level modes, this appears to be the time-weighted average of EPAP and IPAP actually provided.
this - > AddEvent ( new PRS1PressureAverageEvent ( t , value ) ) ;
} else {
// For single-pressure modes, this appears to be the average effective "EPAP" provided by Flex.
//
// Sample data shows this value around 10.3 cmH2O for a prescribed pressure of 12.0 (C-Flex+ 3).
// That's too low for an average pressure over time, but could easily be an average commanded EPAP.
// When flex mode is off, this is exactly the current CPAP set point.
this - > AddEvent ( new PRS1FlexPressureAverageEvent ( t , value ) ) ;
}
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-10-04 22:53:10 +00:00
break ;
case 0x12 : // Snore count per pressure
// Some sessions (with lots of ramps) have multiple of these, each with a
// different pressure. The total snore count across all of them matches the
// total found in the stats event.
if ( data [ pos ] ! = 0 ) {
CHECK_VALUES ( data [ pos ] , 1 , 2 ) ; // 0 = CPAP pressure, 1 = bi-level EPAP, 2 = bi-level IPAP
}
//CHECK_VALUE(data[pos+1], 0x78); // pressure
//CHECK_VALUE(data[pos+2], 1); // 16-bit snore count
//CHECK_VALUE(data[pos+3], 0);
value = ( data [ pos + 2 ] | ( data [ pos + 3 ] < < 8 ) ) ;
this - > AddEvent ( new PRS1SnoresAtPressureEvent ( t , data [ pos ] , data [ pos + 1 ] , value ) ) ;
break ;
default :
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos ) ) ;
2019-11-09 20:09:02 +00:00
ok = false ; // unlike F0V6, we don't know the size of unknown events, so we can't recover
2019-10-04 22:53:10 +00:00
break ;
2011-07-10 14:23:07 +00:00
}
2019-10-04 19:10:08 +00:00
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing event bytes " ;
2011-07-10 14:23:07 +00:00
}
2019-10-04 19:10:08 +00:00
2019-05-25 21:00:44 +00:00
this - > duration = t ;
2012-02-02 03:48:44 +00:00
2019-10-04 19:10:08 +00:00
return ok ;
2011-07-10 14:23:07 +00:00
}
2019-10-22 16:30:42 +00:00
static const QVector < PRS1ParsedEventType > ParsedEventsF0V6 = {
PRS1PressureSetEvent : : TYPE ,
PRS1IPAPSetEvent : : TYPE ,
PRS1EPAPSetEvent : : TYPE ,
PRS1AutoPressureSetEvent : : TYPE ,
PRS1PressurePulseEvent : : TYPE ,
PRS1RERAEvent : : TYPE ,
PRS1ObstructiveApneaEvent : : TYPE ,
PRS1ClearAirwayEvent : : TYPE ,
PRS1HypopneaEvent : : TYPE ,
PRS1FlowLimitationEvent : : TYPE ,
PRS1VibratorySnoreEvent : : TYPE ,
2019-10-30 22:38:13 +00:00
PRS1UnknownDurationEvent : : TYPE ,
2019-10-22 16:30:42 +00:00
PRS1PeriodicBreathingEvent : : TYPE ,
PRS1LargeLeakEvent : : TYPE ,
PRS1TotalLeakEvent : : TYPE ,
PRS1SnoreEvent : : TYPE ,
PRS1PressureAverageEvent : : TYPE ,
2020-01-28 16:26:31 +00:00
PRS1FlexPressureAverageEvent : : TYPE ,
2019-10-22 16:30:42 +00:00
PRS1SnoresAtPressureEvent : : TYPE ,
} ;
2019-10-24 01:10:56 +00:00
// DreamStation family 0 CPAP/APAP machines (400X-700X, 400G-502G)
2019-08-13 21:29:05 +00:00
// Originally derived from F5V3 parsing + (incomplete) F0V234 parsing + sample data
2019-10-09 17:35:02 +00:00
bool PRS1DataChunk : : ParseEventsF0V6 ( )
2019-08-05 00:36:40 +00:00
{
if ( this - > family ! = 0 | | this - > familyVersion ! = 6 ) {
qWarning ( ) < < " ParseEventsF0V6 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
static const int minimum_sizes [ ] = { 2 , 3 , 4 , 3 , 3 , 3 , 3 , 3 , 3 , 2 , 3 , 4 , 3 , 2 , 5 , 5 , 5 , 5 , 4 , 3 , 3 , 3 } ;
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
if ( chunk_size < 1 ) {
// This does occasionally happen.
qDebug ( ) < < this - > sessionid < < " Empty event data " ;
return false ;
}
bool ok = true ;
int pos = 0 , startpos ;
int code , size ;
int t = 0 ;
2019-09-19 20:43:24 +00:00
int elapsed , duration , value ;
2020-01-28 16:26:31 +00:00
bool is_bilevel = false ;
2019-08-05 00:36:40 +00:00
do {
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for event " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
if ( size < minimum_sizes [ code ] ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " too small " < < size < < " < " < < minimum_sizes [ code ] ;
ok = false ;
break ;
}
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " event " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
startpos = pos ;
2019-08-13 21:29:05 +00:00
if ( code ! = 0x12 ) { // This one event has no timestamp
2019-08-05 20:22:22 +00:00
t + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
pos + = 2 ;
}
2019-08-05 00:36:40 +00:00
switch ( code ) {
2019-08-13 21:29:05 +00:00
//case 0x00: // never seen
case 0x01 : // Pressure adjustment
2019-08-05 20:22:22 +00:00
// Matches pressure setting, both initial and when ramp button pressed.
2019-08-12 23:23:47 +00:00
// Based on waveform reports, it looks like the pressure graph is drawn by
2019-08-06 02:37:19 +00:00
// interpolating between these pressure adjustments, by 0.5 cmH2O spaced evenly between
// adjustments. E.g. 6 at 28:11 and 7.3 at 29:05 results in the following dots:
// 6 at 28:11, 6.5 around 28:30, 7.0 around 28:50, 7(.3) at 29:05. That holds until
// subsequent "adjustment" of 7.3 at 30:09 followed by 8.0 at 30:19.
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1PressureSetEvent ( t , data [ pos ] ) ) ;
2019-08-05 00:36:40 +00:00
break ;
2019-08-13 21:29:05 +00:00
case 0x02 : // Pressure adjustment (bi-level)
2019-08-06 20:47:36 +00:00
// See notes above on interpolation.
2019-09-19 18:21:12 +00:00
this - > AddEvent ( new PRS1IPAPSetEvent ( t , data [ pos + 1 ] ) ) ;
this - > AddEvent ( new PRS1EPAPSetEvent ( t , data [ pos ] ) ) ; // EPAP needs to be added second to calculate PS
2020-01-28 16:26:31 +00:00
is_bilevel = true ;
2019-08-06 20:47:36 +00:00
break ;
2019-08-14 03:02:28 +00:00
case 0x03 : // Auto-CPAP starting pressure
// Most of the time this occurs, it's at the start and end of a session with
// the same pressure at both. Occasionally an additional event shows up in the
// middle of a session, and then the pressure at the end matches that.
// In these cases, the new pressure corresponds to the next night's starting
// pressure for auto-CPAP. It does not appear to have any effect on the current
// night's pressure, unless there's a substantial gap between sessions, in
// which case the next session may use the new starting pressure.
//CHECK_VALUE(data[pos], 40);
2019-09-19 20:43:24 +00:00
// TODO: What does this mean in bi-level mode?
2019-10-04 22:53:10 +00:00
// See F0V4 event 3 for comparison. TODO: See if there's an Opti-Start label on F0V6 reports.
2019-09-19 20:43:24 +00:00
this - > AddEvent ( new PRS1AutoPressureSetEvent ( t , data [ pos ] ) ) ;
2019-08-05 00:36:40 +00:00
break ;
case 0x04 : // Pressure Pulse
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ; // TODO: is this a duration?
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1PressurePulseEvent ( t , duration ) ) ;
break ;
2019-08-05 02:09:42 +00:00
case 0x05 : // RERA
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ; // based on sample waveform, the RERA is over after this
2019-08-05 02:09:42 +00:00
this - > AddEvent ( new PRS1RERAEvent ( t - elapsed , 0 ) ) ;
break ;
case 0x06 : // Obstructive Apnea
2019-08-05 00:36:40 +00:00
// OA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1ObstructiveApneaEvent ( t - elapsed , 0 ) ) ;
break ;
2019-08-05 02:09:42 +00:00
case 0x07 : // Clear Airway Apnea
2019-08-05 00:36:40 +00:00
// CA events are instantaneous flags with no duration: reviewing waveforms
// shows that the time elapsed between the flag and reporting often includes
// non-apnea breathing.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1ClearAirwayEvent ( t - elapsed , 0 ) ) ;
break ;
2019-08-13 21:29:05 +00:00
//case 0x08: // never seen
//case 0x09: // never seen
//case 0x0a: // Hypopnea, see 0x15
2019-08-05 02:09:42 +00:00
case 0x0b : // Hypopnea
2019-08-13 21:29:05 +00:00
// TODO: How is this hypopnea different from events 0xa, 0x14 and 0x15?
2019-08-05 00:36:40 +00:00
// TODO: What is the first byte?
2019-08-20 15:38:55 +00:00
//data[pos+0]; // unknown first byte?
elapsed = data [ pos + 1 ] ; // based on sample waveform, the hypopnea is over after this
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1HypopneaEvent ( t - elapsed , 0 ) ) ;
break ;
2019-08-06 02:37:19 +00:00
case 0x0c : // Flow Limitation
2019-08-05 00:36:40 +00:00
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating flow limitations ourselves. Flow limitations aren't
// as obvious as OA/CA when looking at a waveform.
2019-08-20 15:38:55 +00:00
elapsed = data [ pos ] ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1FlowLimitationEvent ( t - elapsed , 0 ) ) ;
break ;
2019-08-06 02:37:19 +00:00
case 0x0d : // Vibratory Snore
2019-08-05 00:36:40 +00:00
// VS events are instantaneous flags with no duration, drawn on the official waveform.
// The current thinking is that these are the snores that cause a change in auto-titrating
2019-08-13 21:29:05 +00:00
// pressure. The snoring statistics below seem to be a total count. It's unclear whether
2019-08-05 00:36:40 +00:00
// the trigger for pressure change is severity or count or something else.
// no data bytes
this - > AddEvent ( new PRS1VibratorySnoreEvent ( t , 0 ) ) ;
break ;
2019-08-05 20:22:22 +00:00
case 0x0e : // ???
// 5 bytes like PB and LL, but what is it?
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ; // this looks like a 16-bit value, so may be duration like PB?
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ; // this is always 60 seconds unless it's at the end, so it seems like elapsed
2019-08-05 20:22:22 +00:00
CHECK_VALUES ( elapsed , 60 , 0 ) ;
2019-09-19 20:43:24 +00:00
this - > AddEvent ( new PRS1UnknownDurationEvent ( t - elapsed - duration , duration ) ) ;
2019-08-05 20:22:22 +00:00
break ;
2019-08-05 02:09:42 +00:00
case 0x0f : // Periodic Breathing
2019-08-05 00:36:40 +00:00
// PB events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1PeriodicBreathingEvent ( t - elapsed - duration , duration ) ) ;
break ;
2019-08-05 02:09:42 +00:00
case 0x10 : // Large Leak
2019-08-05 00:36:40 +00:00
// LL events are reported some time after they conclude, and they do have a reported duration.
duration = 2 * ( data [ pos ] | ( data [ pos + 1 ] < < 8 ) ) ;
2019-08-20 15:38:55 +00:00
elapsed = data [ pos + 2 ] ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1LargeLeakEvent ( t - elapsed - duration , duration ) ) ;
break ;
2019-08-13 21:29:05 +00:00
case 0x11 : // Statistics
2019-08-20 15:38:55 +00:00
this - > AddEvent ( new PRS1TotalLeakEvent ( t , data [ pos ] ) ) ;
this - > AddEvent ( new PRS1SnoreEvent ( t , data [ pos + 1 ] ) ) ;
2020-01-28 16:26:31 +00:00
value = data [ pos + 2 ] ;
if ( is_bilevel ) {
// For bi-level modes, this appears to be the time-weighted average of EPAP and IPAP actually provided.
this - > AddEvent ( new PRS1PressureAverageEvent ( t , value ) ) ;
} else {
// For single-pressure modes, this appears to be the average effective "EPAP" provided by Flex.
//
// Sample data shows this value around 10.3 cmH2O for a prescribed pressure of 12.0 (C-Flex+ 3).
// That's too low for an average pressure over time, but could easily be an average commanded EPAP.
// When flex mode is off, this is exactly the current CPAP set point.
this - > AddEvent ( new PRS1FlexPressureAverageEvent ( t , value ) ) ;
}
2020-01-06 16:59:15 +00:00
this - > AddEvent ( new PRS1IntervalBoundaryEvent ( t ) ) ;
2019-08-13 21:29:05 +00:00
break ;
case 0x12 : // Snore count per pressure
// Some sessions (with lots of ramps) have multiple of these, each with a
// different pressure. The total snore count across all of them matches the
// total found in the stats event.
if ( data [ pos ] ! = 0 ) {
CHECK_VALUES ( data [ pos ] , 1 , 2 ) ; // 0 = CPAP pressure, 1 = bi-level EPAP, 2 = bi-level IPAP
}
//CHECK_VALUE(data[pos+1], 0x78); // pressure
//CHECK_VALUE(data[pos+2], 1); // 16-bit snore count
//CHECK_VALUE(data[pos+3], 0);
2019-09-19 20:43:24 +00:00
value = ( data [ pos + 2 ] | ( data [ pos + 3 ] < < 8 ) ) ;
this - > AddEvent ( new PRS1SnoresAtPressureEvent ( t , data [ pos ] , data [ pos + 1 ] , value ) ) ;
2019-08-13 21:29:05 +00:00
break ;
//case 0x13: // never seen
2019-08-06 02:37:19 +00:00
case 0x0a : // Hypopnea
// TODO: Why does this hypopnea have a different event code?
// fall through
2019-10-04 22:53:10 +00:00
case 0x14 : // Hypopnea, new to F0V6
2019-08-05 00:36:40 +00:00
// TODO: Why does this hypopnea have a different event code?
// fall through
2019-10-04 22:53:10 +00:00
case 0x15 : // Hypopnea, new to F0V6
2019-08-05 00:36:40 +00:00
// TODO: We should revisit whether this is elapsed or duration once (if)
// we start calculating hypopneas ourselves. Their official definition
// is 40% reduction in flow lasting at least 10s.
2019-08-20 15:38:55 +00:00
duration = data [ pos ] ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1HypopneaEvent ( t - duration , 0 ) ) ;
break ;
default :
2020-03-23 00:00:09 +00:00
DUMP_EVENT ( ) ;
UNEXPECTED_VALUE ( code , " known event code " ) ;
2019-08-05 00:36:40 +00:00
this - > AddEvent ( new PRS1UnknownDataEvent ( m_data , startpos - 1 , size + 1 ) ) ;
break ;
}
pos = startpos + size ;
} while ( ok & & pos < chunk_size ) ;
this - > duration = t ;
return ok ;
}
2019-10-22 16:30:42 +00:00
// TODO: This really should be in some kind of class hierarchy, once we figure out
// the right one.
const QVector < PRS1ParsedEventType > & GetSupportedEvents ( const PRS1DataChunk * chunk )
{
static const QVector < PRS1ParsedEventType > none ;
switch ( chunk - > family ) {
case 0 :
switch ( chunk - > familyVersion ) {
case 2 : return ParsedEventsF0V23 ; break ;
case 3 : return ParsedEventsF0V23 ; break ;
case 4 : return ParsedEventsF0V4 ; break ;
case 6 : return ParsedEventsF0V6 ; break ;
}
break ;
case 3 :
switch ( chunk - > familyVersion ) {
case 3 : return ParsedEventsF3V3 ; break ;
case 6 : return ParsedEventsF3V6 ; break ;
}
break ;
case 5 :
switch ( chunk - > familyVersion ) {
case 0 : return ParsedEventsF5V0 ; break ;
case 1 : return ParsedEventsF5V1 ; break ;
case 2 : return ParsedEventsF5V2 ; break ;
case 3 : return ParsedEventsF5V3 ; break ;
}
break ;
}
qWarning ( ) < < " Missing supported event list for family " < < chunk - > family < < " version " < < chunk - > familyVersion ;
return none ;
}
2019-08-29 01:30:25 +00:00
CPAPMode PRS1Import : : importMode ( int prs1mode )
{
CPAPMode mode ;
switch ( prs1mode ) {
case PRS1_MODE_CPAP : mode = MODE_CPAP ; break ;
case PRS1_MODE_CPAPCHECK : mode = MODE_CPAP ; break ; // "CPAP-Check" in report, but seems like CPAP
case PRS1_MODE_AUTOCPAP : mode = MODE_APAP ; break ;
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 ;
case PRS1_MODE_S : mode = MODE_BILEVEL_FIXED ; break ; // TODO
case PRS1_MODE_ST : mode = MODE_BILEVEL_FIXED ; break ; // TODO, pressure seems variable
case PRS1_MODE_PC : mode = MODE_AVAPS ; break ; // TODO, maybe only PC - AVAPS mode
default : mode = MODE_UNKNOWN ; break ;
}
// TODO: fixed vs. variable PS seems to be independent from ventilator mode, for example
// S/T can be fixed (single IPAP pressure) or variable (IPAP min/max).
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 :
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 ;
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 :
// Don't bother importing these for bricks, because they're always locked off.
CHECK_VALUE ( e - > m_value , 0 ) ;
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 ;
}
bool PRS1DataChunk : : ParseCompliance ( void )
{
2019-06-06 20:08:40 +00:00
switch ( this - > family ) {
case 0 :
if ( this - > familyVersion = = 6 ) {
return this - > ParseComplianceF0V6 ( ) ;
} else if ( this - > familyVersion = = 2 | | this - > familyVersion = = 3 ) {
return this - > ParseComplianceF0V23 ( ) ;
}
default :
;
}
qWarning ( ) < < " unexpected family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
bool PRS1DataChunk : : ParseComplianceF0V23 ( void )
{
if ( this - > family ! = 0 | | ( this - > familyVersion ! = 2 & & this - > familyVersion ! = 3 ) ) {
qWarning ( ) < < " ParseComplianceF0V23 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
2019-05-31 20:58:58 +00:00
return false ;
}
2019-10-04 16:08:16 +00:00
// All sample machines with FamilyVersion 3 in the properties.txt file have familyVersion 2 in their .001/.002/.005 files!
// We should flag an actual familyVersion 3 file if we ever encounter one!
2019-06-06 20:08:40 +00:00
CHECK_VALUE ( this - > familyVersion , 2 ) ;
2019-05-26 18:17:58 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
2019-10-24 15:44:47 +00:00
int chunk_size = this - > m_data . size ( ) ;
static const int minimum_sizes [ ] = { 0xd , 5 , 2 , 2 } ;
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
// NOTE: These are fixed sizes, but are called minimum to more closely match the F0V6 parser.
2014-09-01 11:39:38 +00:00
2019-10-24 15:44:47 +00:00
bool ok = true ;
int pos = 0 ;
int code , size , delta ;
int tt = 0 ;
2014-09-04 02:17:59 +00:00
do {
2019-10-24 15:44:47 +00:00
code = data [ pos + + ] ;
// There is no hblock prior to F0V6.
size = 0 ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
size = minimum_sizes [ code ] ;
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
2014-09-04 02:17:59 +00:00
break ;
}
2019-10-24 15:44:47 +00:00
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUES ( data [ pos ] , 1 , 0 ) ; // usually 1, occasionally 0, no visible difference in report
// F0V23 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
ok = ParseSettingsF0V23 ( data , 0x0e ) ;
// Compliance doesn't have pressure set events following settings like summary does.
break ;
case 2 : // Mask On
delta = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
if ( tt = = 0 ) {
CHECK_VALUE ( delta , 0 ) ; // we've never seen the initial MaskOn have any delta
} else {
if ( delta % 60 ) UNEXPECTED_VALUE ( delta , " even minutes " ) ; // mask-off events seem to be whole minutes?
}
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
// no per-slice humidifer settings as in F0V6
break ;
case 3 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
// Compliance doesn't record any stats after mask-off like summary does.
break ;
case 1 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
// also seems to be a trailing 01 00 81 after the slices?
CHECK_VALUES ( data [ pos + 2 ] , 1 , 0 ) ; // usually 1, occasionally 0, no visible difference in report
//CHECK_VALUE(data[pos+3], 0); // sometimes 1, 2, or 5, no visible difference in report
//CHECK_VALUES(data[pos+4], 0x81, 0x80); // seems to be humidifier setting at end of session
if ( data [ pos + 4 ] & & ( ( ( data [ pos + 4 ] & 0x80 ) = = 0 ) | | ( data [ pos + 4 ] & 0x07 ) > 5 ) ) {
UNEXPECTED_VALUE ( data [ pos + 4 ] , " valid humidifier setting " ) ;
}
break ;
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
2019-06-03 02:05:10 +00:00
}
2019-10-24 15:44:47 +00:00
pos + = size ;
} while ( ok & & pos < chunk_size ) ;
if ( ok & & pos ! = chunk_size ) {
2019-05-31 20:58:58 +00:00
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing bytes " ;
}
2019-05-26 18:17:58 +00:00
this - > duration = tt ;
2014-09-01 11:39:38 +00:00
2019-10-24 15:44:47 +00:00
return ok ;
2014-05-31 21:25:07 +00:00
}
2011-12-10 12:14:48 +00:00
2019-05-26 01:57:29 +00:00
bool PRS1DataChunk : : ParseSummaryF0V23 ( )
2014-08-03 13:00:13 +00:00
{
2019-06-04 16:18:20 +00:00
if ( this - > family ! = 0 | | ( this - > familyVersion ! = 2 & & this - > familyVersion ! = 3 ) ) {
qWarning ( ) < < " ParseSummaryF0V23 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
2019-10-04 16:08:16 +00:00
// All sample machines with FamilyVersion 3 in the properties.txt file have familyVersion 2 in their .001/.002/.005 files!
// We should flag an actual familyVersion 3 file if we ever encounter one!
CHECK_VALUE ( this - > familyVersion , 2 ) ;
2019-09-25 01:07:10 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
static const int minimum_sizes [ ] = { 0xf , 5 , 2 , 0x21 } ;
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
// NOTE: These are fixed sizes, but are called minimum to more closely match the F0V6 parser.
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-09-25 01:07:10 +00:00
code = data [ pos + + ] ;
// There is no hblock prior to F0V6.
size = 0 ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
size = minimum_sizes [ code ] ;
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
2019-09-24 20:27:27 +00:00
break ;
}
2019-09-25 01:07:10 +00:00
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUES ( data [ pos ] & 0xF0 , 0x60 , 0x70 ) ; // TODO: what are these?
if ( ( data [ pos ] & 0x0F ) ! = 1 ) { // This is the most frequent value.
CHECK_VALUES ( data [ pos ] & 0x0F , 3 , 0 ) ; // TODO: what are these? 0 seems to be related to errors.
}
// F0V23 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
ok = ParseSettingsF0V23 ( data , 0x0e ) ;
// TODO: register these as pressure set events
//CHECK_VALUES(data[0x0e], ramp_pressure, min_pressure); // initial CPAP/EPAP, can be minimum pressure or ramp, or whatever auto decides to use
//if (cpapmode == PRS1_MODE_BILEVEL) { // initial IPAP for bilevel modes
// CHECK_VALUE(data[0x0f], max_pressure);
//} else if (cpapmode == PRS1_MODE_AUTOBILEVEL) {
// CHECK_VALUE(data[0x0f], min_pressure + 20);
//}
break ;
case 2 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
// no per-slice humidifer settings as in F0V6
break ;
case 3 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
// F0V23 doesn't have a separate stats record like F0V6 does, the stats just follow the MaskOff data.
// These are 0x22 bytes in a summary vs. 3 bytes in compliance data
// TODO: What are these values?
break ;
case 1 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
// seems to be trailing 01 [01 or 02] 83 after equipment off?
if ( data [ pos + 2 ] ! = 1 ) { // This is the usual value.
CHECK_VALUES ( data [ pos + 2 ] , 0 , 3 ) ; // 0 seems to be related to errors, 3 seen after 90 sec large leak before turning off?
}
//CHECK_VALUES(data[pos+3], 0, 1); // TODO: may be related to ramp? 1-5 seems to have a ramp start or two
//CHECK_VALUES(data[pos+4], 0x81, 0x80); // seems to be humidifier setting at end of session
2020-01-10 17:57:18 +00:00
ParseHumidifierSetting50Series ( data [ pos + 4 ] ) ;
2019-09-25 01:07:10 +00:00
break ;
2019-09-30 19:46:26 +00:00
/*
2019-09-25 01:07:10 +00:00
case 5 : // Unknown, but occasionally encountered
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 1 ) ; // and the only record in the session.
ok = false ;
break ;
2019-09-30 19:46:26 +00:00
*/
case 6 : // Cleared?
// Appears in the very first session when that session number is > 1.
// Presumably previous sessions were cleared out.
// TODO: add an internal event for this.
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 1 ) ; // and the only record in the session.
if ( this - > sessionid = = 1 ) UNEXPECTED_VALUE ( this - > sessionid , " >1 " ) ;
break ;
2019-09-25 01:07:10 +00:00
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
2019-09-24 20:27:27 +00:00
}
2019-09-25 01:07:10 +00:00
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-09-25 01:07:10 +00:00
if ( ok & & pos ! = chunk_size ) {
2019-09-24 20:27:27 +00:00
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing bytes " ;
}
this - > duration = tt ;
2019-09-29 00:44:18 +00:00
return ok ;
2019-09-24 20:27:27 +00:00
}
bool PRS1DataChunk : : ParseSettingsF0V23 ( const unsigned char * data , int /*size*/ )
{
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2014-08-03 13:00:13 +00:00
switch ( data [ 0x02 ] ) { // PRS1 mode // 0 = CPAP, 2 = APAP
case 0x00 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_CPAP ;
2014-08-03 13:00:13 +00:00
break ;
case 0x01 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_BILEVEL ;
2014-08-03 13:00:13 +00:00
break ;
case 0x02 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_AUTOCPAP ;
2014-08-03 13:00:13 +00:00
break ;
case 0x03 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_AUTOBILEVEL ;
2019-06-05 03:19:35 +00:00
break ;
default :
2020-03-23 00:00:09 +00:00
UNEXPECTED_VALUE ( data [ 0x02 ] , " known device mode " ) ;
break ;
2014-08-03 13:00:13 +00:00
}
2019-06-04 16:18:20 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
2014-08-03 13:00:13 +00:00
2019-05-26 01:57:29 +00:00
int min_pressure = data [ 0x03 ] ;
int max_pressure = data [ 0x04 ] ;
2019-06-05 03:19:35 +00:00
int ps = data [ 0x05 ] ; // max pressure support (for variable), seems to be zero otherwise
2014-08-03 13:00:13 +00:00
2019-08-29 01:30:25 +00:00
if ( cpapmode = = PRS1_MODE_CPAP ) {
2019-05-26 01:57:29 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE , min_pressure ) ) ;
2019-06-05 03:19:35 +00:00
CHECK_VALUE ( max_pressure , 0 ) ;
CHECK_VALUE ( ps , 0 ) ;
2019-08-29 01:30:25 +00:00
} else if ( cpapmode = = PRS1_MODE_AUTOCPAP ) {
2019-05-26 01:57:29 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MIN , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MAX , max_pressure ) ) ;
2019-06-05 03:19:35 +00:00
CHECK_VALUE ( ps , 0 ) ;
2019-08-29 01:30:25 +00:00
} else if ( cpapmode = = PRS1_MODE_BILEVEL ) {
2019-05-26 01:57:29 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP , max_pressure ) ) ;
2019-06-05 03:19:35 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS , max_pressure - min_pressure ) ) ;
CHECK_VALUE ( ps , 0 ) ; // this seems to be unused on fixed bilevel
2019-08-29 01:30:25 +00:00
} else if ( cpapmode = = PRS1_MODE_AUTOBILEVEL ) {
2019-05-26 21:43:10 +00:00
int min_ps = 20 ; // 2.0 cmH2O
2019-05-26 01:57:29 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MIN , min_pressure ) ) ;
2019-06-05 03:19:35 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MAX , max_pressure - min_ps ) ) ; // TODO: not yet confirmed
2019-05-26 01:57:29 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MIN , min_pressure + min_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , max_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MIN , min_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MAX , ps ) ) ;
2014-08-03 13:00:13 +00:00
}
int ramp_time = data [ 0x06 ] ;
2019-05-26 01:57:29 +00:00
int ramp_pressure = data [ 0x07 ] ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , ramp_time ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , ramp_pressure ) ) ;
2014-08-03 13:00:13 +00:00
2019-06-04 16:18:20 +00:00
quint8 flex = data [ 0x08 ] ;
2020-01-12 21:53:43 +00:00
this - > ParseFlexSettingF0V234 ( flex , cpapmode ) ;
2019-06-04 16:18:20 +00:00
int humid = data [ 0x09 ] ;
2020-01-10 17:57:18 +00:00
this - > ParseHumidifierSetting50Series ( humid , true ) ;
2019-06-04 16:18:20 +00:00
2014-08-03 13:00:13 +00:00
// Tubing lock has no setting byte
// Menu Options
2020-03-23 16:59:06 +00:00
bool mask_resist_on = ( ( data [ 0x0a ] & 0x40 ) ! = 0 ) ; // System One Resistance Status bit
int mask_resist_setting = data [ 0x0a ] & 7 ; // System One Resistance setting value
CHECK_VALUE ( mask_resist_on , mask_resist_setting > 0 ) ; // Confirm that we can ignore the status bit.
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , ( data [ 0x0a ] & 0x80 ) ! = 0 ) ) ; // System One Resistance Lock Setting, only seen on bricks
2019-06-05 03:19:35 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HOSE_DIAMETER , ( data [ 0x0a ] & 0x08 ) ? 15 : 22 ) ) ; // TODO: unconfirmed
2020-03-23 16:59:06 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , mask_resist_setting ) ) ;
2019-06-04 16:18:20 +00:00
CHECK_VALUE ( data [ 0x0a ] & ( 0x20 | 0x10 ) , 0 ) ;
CHECK_VALUE ( data [ 0x0b ] , 1 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_ON , ( data [ 0x0c ] & 0x40 ) ! = 0 ) ) ;
2019-05-26 01:57:29 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_OFF , ( data [ 0x0c ] & 0x10 ) ! = 0 ) ) ;
2019-06-04 16:18:20 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_ALERT , ( data [ 0x0c ] & 0x04 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_SHOW_AHI , ( data [ 0x0c ] & 0x02 ) ! = 0 ) ) ;
CHECK_VALUE ( data [ 0x0c ] & ( 0xA0 | 0x09 ) , 0 ) ;
2014-08-03 13:00:13 +00:00
2019-06-04 16:18:20 +00:00
CHECK_VALUE ( data [ 0x0d ] , 0 ) ;
2014-08-03 13:00:13 +00:00
return true ;
}
2019-05-26 01:25:56 +00:00
2019-09-24 20:27:27 +00:00
bool PRS1DataChunk : : ParseSettingsF0V4 ( const unsigned char * data , int /*size*/ )
2019-05-26 01:25:56 +00:00
{
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2014-08-03 15:24:58 +00:00
2019-09-25 15:17:16 +00:00
switch ( data [ 0x02 ] ) { // PRS1 mode
2014-08-03 15:24:58 +00:00
case 0x00 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_CPAP ;
2014-08-03 15:24:58 +00:00
break ;
case 0x20 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_BILEVEL ;
2014-08-03 15:24:58 +00:00
break ;
case 0x40 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_AUTOCPAP ;
2014-08-03 15:24:58 +00:00
break ;
case 0x60 :
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_AUTOBILEVEL ;
2019-09-25 15:17:16 +00:00
break ;
2019-09-26 16:37:01 +00:00
case 0x80 :
cpapmode = PRS1_MODE_AUTOTRIAL ; // Auto-Trial TODO: where is duration?
break ;
case 0xA0 :
cpapmode = PRS1_MODE_CPAPCHECK ;
break ;
2019-09-25 15:17:16 +00:00
default :
2020-03-23 00:00:09 +00:00
UNEXPECTED_VALUE ( data [ 0x02 ] , " known device mode " ) ;
2019-09-25 15:17:16 +00:00
break ;
2014-08-03 15:24:58 +00:00
}
2019-09-25 15:17:16 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
2019-05-26 01:25:56 +00:00
int min_pressure = data [ 0x03 ] ;
int max_pressure = data [ 0x04 ] ;
int min_ps = data [ 0x05 ] ; // pressure support
int max_ps = data [ 0x06 ] ; // pressure support
2014-08-03 15:24:58 +00:00
2019-08-29 01:30:25 +00:00
if ( cpapmode = = PRS1_MODE_CPAP ) {
2019-05-26 01:25:56 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE , min_pressure ) ) ;
2019-09-25 15:17:16 +00:00
CHECK_VALUE ( max_pressure , 0 ) ;
CHECK_VALUE ( min_ps , 0 ) ;
CHECK_VALUE ( max_ps , 0 ) ;
2019-09-26 16:37:01 +00:00
} else if ( cpapmode = = PRS1_MODE_AUTOCPAP | | cpapmode = = PRS1_MODE_AUTOTRIAL ) {
2019-05-26 01:25:56 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MIN , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MAX , max_pressure ) ) ;
2019-09-25 15:17:16 +00:00
CHECK_VALUE ( min_ps , 0 ) ;
CHECK_VALUE ( max_ps , 0 ) ;
2019-09-26 16:37:01 +00:00
} else if ( cpapmode = = PRS1_MODE_CPAPCHECK ) {
2019-09-28 22:23:18 +00:00
// Sometimes the CPAP pressure is stored in max_ps instead of min_ps, not sure why.
if ( min_ps = = 0 ) {
if ( max_ps = = 0 ) UNEXPECTED_VALUE ( max_ps , " nonzero " ) ;
min_ps = max_ps ;
} else {
CHECK_VALUE ( max_ps , 0 ) ;
}
2019-09-26 16:37:01 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE , min_ps ) ) ;
2020-03-23 00:00:09 +00:00
// TODO: Once OSCAR can handle more modes, we can include these settings; right now including
// these settings makes it think this is AutoCPAP.
//this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure));
//this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure));
2019-08-29 01:30:25 +00:00
} else if ( cpapmode = = PRS1_MODE_BILEVEL ) {
2019-05-26 01:25:56 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP , max_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS , max_pressure - min_pressure ) ) ;
2019-09-25 15:17:16 +00:00
CHECK_VALUE ( min_ps , 0 ) ; // this seems to be unused on fixed bilevel
CHECK_VALUE ( max_ps , 0 ) ; // this seems to be unused on fixed bilevel
2019-08-29 01:30:25 +00:00
} else if ( cpapmode = = PRS1_MODE_AUTOBILEVEL ) {
2019-05-26 01:25:56 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MIN , min_pressure ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MAX , max_pressure - min_ps ) ) ;
2019-05-26 01:25:56 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MIN , min_pressure + min_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , max_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MIN , min_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MAX , max_ps ) ) ;
2014-08-03 15:24:58 +00:00
}
2019-09-28 22:23:18 +00:00
CHECK_VALUES ( data [ 0x07 ] , 0 , 0x20 ) ; // 0x20 seems to be Opti-Start
2014-08-03 15:24:58 +00:00
2014-08-11 01:44:25 +00:00
int ramp_time = data [ 0x08 ] ;
2019-05-26 01:25:56 +00:00
int ramp_pressure = data [ 0x09 ] ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , ramp_time ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , ramp_pressure ) ) ;
2014-08-11 01:44:25 +00:00
2019-09-25 15:17:16 +00:00
quint8 flex = data [ 0x0a ] ;
2020-01-12 21:53:43 +00:00
this - > ParseFlexSettingF0V234 ( flex , cpapmode ) ;
2019-09-25 15:17:16 +00:00
2020-01-10 21:44:00 +00:00
this - > ParseHumidifierSetting60Series ( data [ 0x0b ] , data [ 0x0c ] , true ) ;
2019-09-28 23:31:37 +00:00
2019-09-29 00:44:18 +00:00
int resist_level = ( data [ 0x0d ] > > 3 ) & 7 ; // 0x18 resist=3, 0x11 resist=2
CHECK_VALUE ( data [ 0x0d ] & 0x20 , 0 ) ; // never seen, but would clarify whether above mask is correct
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , ( data [ 0x0d ] & 0x40 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , resist_level ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HOSE_DIAMETER , ( data [ 0x0d ] & 0x01 ) ? 15 : 22 ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TUBING_LOCK , ( data [ 0x0d ] & 0x02 ) ! = 0 ) ) ;
2019-09-29 00:44:18 +00:00
CHECK_VALUE ( data [ 0x0d ] & ( 0x80 | 0x04 ) , 0 ) ;
CHECK_VALUE ( data [ 0x0e ] , 1 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_ON , ( data [ 0x0f ] & 0x40 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_OFF , ( data [ 0x0f ] & 0x10 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_ALERT , ( data [ 0x0f ] & 0x04 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_SHOW_AHI , ( data [ 0x0f ] & 0x02 ) ! = 0 ) ) ;
CHECK_VALUE ( data [ 0x0f ] & ( 0xA0 | 0x08 ) , 0 ) ;
2020-03-23 00:00:09 +00:00
//CHECK_VALUE(data[0x0f] & 0x01, 0); // TODO: What is bit 1? It's sometimes set.
2019-09-29 00:44:18 +00:00
2019-10-01 15:29:31 +00:00
CHECK_VALUE ( data [ 0x10 ] , 0 ) ;
if ( cpapmode = = PRS1_MODE_AUTOTRIAL ) {
CHECK_VALUE ( data [ 0x11 ] , 7 ) ; // 7-day duration?
} else {
CHECK_VALUE ( data [ 0x11 ] , 0 ) ;
}
2019-09-29 00:44:18 +00:00
return true ;
}
2019-09-26 16:37:01 +00:00
2019-09-29 00:44:18 +00:00
2020-01-10 21:44:00 +00:00
// F0V4 confirmed:
2020-01-10 15:49:08 +00:00
// B3 0A = HT=5, H=3, HT
// A3 0A = HT=5, H=2, HT
// 33 0A = HT=4, H=3, HT
// 23 4A = HT=4, H=2, HT
// B3 09 = HT=3, H=3, HT
// A4 09 = HT=3, H=2, HT
// A3 49 = HT=3, H=2, HT
// 22 09 = HT=2, H=2, HT
// 33 09 = HT=2, H=3, HT
// 21 09 = HT=2, H=2, HT
// 13 09 = HT=2, H=1, HT
// B5 08 = HT=1, H=3, HT
// 03 08 = HT=off, HT; data=tube t=0,h=0
// 05 24 = H=5, S1
// 95 06 = H=5, S1
// 95 05 = H=5, S1
// 94 05 = H=4, S1
// 04 24 = H=4, S1
2020-01-10 21:44:00 +00:00
// A3 05 = H=3, S1
2020-01-10 15:49:08 +00:00
// 92 05 = H=2, S1
// A2 05 = H=2, S1
// 01 24 = H=1, S1
// 90 05 = H=off, S1
// 30 05 = H=off, S1
// 95 41 = H=5, Classic
2020-01-10 21:44:00 +00:00
// A4 61 = H=4, Classic
2020-01-10 15:49:08 +00:00
// A3 61 = H=3, Classic
2020-01-10 21:44:00 +00:00
// A2 61 = H=2, Classic
2020-01-10 15:49:08 +00:00
// A1 61 = H=1, Classic
// 90 41 = H=Off, Classic; data=classic h=0
// 94 11 = H=3, S1, no data [note that bits encode H=4, so no data falls back to H=3]
// 93 11 = H=3, S1, no data
// 04 30 = H=3, S1, no data
// F5V1 confirmed:
// A0 4A = HT=5, H=2, HT
// B1 09 = HT=3, H=3, HT
2020-01-10 21:44:00 +00:00
// 91 09 = HT=3, H=1, HT
2020-01-10 15:49:08 +00:00
// 32 09 = HT=2, H=3, HT
2020-01-10 21:44:00 +00:00
// B2 08 = HT=1, H=3, HT
// 00 48 = HT=off, data=tube t=0,h=0
// 95 05 = H=5, S1
// 94 05 = H=4, S1
// 93 05 = H=3, S1
// 92 05 = H=2, S1
// 91 05 = H=1, S1
// 90 05 = H=Off, S1
2020-01-12 00:23:43 +00:00
// 95 41 = H=5, Classic
// 94 41 = H=4, Classic
// 93 41 = H=3, Classic
// 92 41 = H=2, Classic
2020-01-10 21:44:00 +00:00
// 01 60 = H=1, Classic
// 00 60 = H=Off, Classic
// 00 70 = H=3, S1, no data [no data ignores Classic mode, H bits, falls back to S1 H=3]
2020-01-10 15:49:08 +00:00
// F5V2 confirmed:
// 00 48 = HT=off, data=tube t=0,h=0
// 93 09 = HT=3, H=1, HT
2020-01-10 21:44:00 +00:00
// 00 10 = H=3, S1, no data
2020-01-10 15:49:08 +00:00
2020-01-10 21:44:00 +00:00
// XX XX = 60-Series Humidifier bytes
2020-01-10 15:49:08 +00:00
// 7 = humidity level without tube [on tube disconnect / system one with 22mm hose / classic] : 0 = humidifier off
2019-09-29 00:44:18 +00:00
// 8 = [never seen]
// 3 = humidity level with tube
// 4 = maybe part of humidity level? [never seen]
// 8 3 = tube temperature (high bit of humid 1 is low bit of temp)
2020-01-10 15:49:08 +00:00
// 4 = "System One" mode (valid even when humidifier is off)
// 8 = heated tube present
// 10 = no data in chart, maybe no humidifier attached? Seems to fall back on System One = 3 despite other (humidity level and S1) bits.
2019-09-29 00:44:18 +00:00
// 20 = unknown, something tube related since whenever it's set tubepresent is false
2020-01-10 15:49:08 +00:00
// 40 = "Classic" mode (valid even when humidifier is off, ignored when heated tube is present)
2019-09-29 00:44:18 +00:00
// 80 = [never seen]
2020-01-10 21:44:00 +00:00
void PRS1DataChunk : : ParseHumidifierSetting60Series ( unsigned char humid1 , unsigned char humid2 , bool add_setting )
2019-09-29 00:44:18 +00:00
{
2019-09-28 22:23:18 +00:00
int humidlevel = humid1 & 7 ; // Ignored when heated tube is present: humidifier setting on tube disconnect is always reported as 3
if ( humidlevel > 5 ) UNEXPECTED_VALUE ( humidlevel , " <= 5 " ) ;
2019-09-26 16:37:01 +00:00
CHECK_VALUE ( humid1 & 8 , 0 ) ; // never seen
2019-09-28 22:23:18 +00:00
int tubehumidlevel = ( humid1 > > 4 ) & 7 ; // This mask is a best guess based on other masks.
if ( tubehumidlevel > 5 ) UNEXPECTED_VALUE ( tubehumidlevel , " <= 5 " ) ;
CHECK_VALUE ( tubehumidlevel & 4 , 0 ) ; // never seen, but would clarify whether above mask is correct
int tubetemp = ( humid1 > > 7 ) | ( ( humid2 & 3 ) < < 1 ) ;
if ( tubetemp > 5 ) UNEXPECTED_VALUE ( tubetemp , " <= 5 " ) ;
2019-09-26 16:37:01 +00:00
CHECK_VALUE ( humid2 & 0x80 , 0 ) ; // never seen
2019-09-28 22:23:18 +00:00
bool humidclassic = ( humid2 & 0x40 ) ! = 0 ; // Set on classic mode reports; evidently ignored (sometimes set!) when tube is present
//bool no_tube? = (humid2 & 0x20) != 0; // Something tube related: whenever it is set, tube is never present (inverse is not true)
bool no_data = ( humid2 & 0x10 ) ! = 0 ; // As described in chart, settings still show up
2019-09-26 16:37:01 +00:00
int tubepresent = ( humid2 & 0x08 ) ! = 0 ;
2019-09-28 22:23:18 +00:00
bool humidsystemone = ( humid2 & 0x04 ) ! = 0 ; // Set on "System One" humidification mode reports when tubepresent is false
2020-01-10 15:49:08 +00:00
// When no_data, reports always say "System One" with humidity level 3, regardless of humidlevel and humidsystemone
2019-09-28 22:23:18 +00:00
if ( humidsystemone + tubepresent + no_data = = 0 ) CHECK_VALUE ( humidclassic , true ) ; // Always set when everything else is off
if ( humidsystemone + tubepresent + no_data > 1 ) UNEXPECTED_VALUE ( humid2 , " one bit set " ) ; // Only one of these ever seems to be set at a time
if ( tubepresent & & tubetemp = = 0 ) CHECK_VALUE ( tubehumidlevel , 0 ) ; // When the heated tube is off, tube humidity seems to be 0
2020-01-10 15:49:08 +00:00
if ( tubepresent ) humidclassic = false ; // Classic mode bit is evidently ignored when tube is present
2020-01-10 21:44:00 +00:00
if ( no_data ) humidclassic = false ; // Classic mode bit is evidently ignored when tube is present
2020-01-10 15:49:08 +00:00
2019-09-28 23:31:37 +00:00
//qWarning() << this->sessionid << (humidclassic ? "C" : ".") << (humid2 & 0x20 ? "?" : ".") << (tubepresent ? "T" : ".") << (no_data ? "X" : ".") << (humidsystemone ? "1" : ".");
2019-09-28 22:23:18 +00:00
/*
if ( tubepresent ) {
if ( tubetemp ) {
qWarning ( ) < < this - > sessionid < < " tube temp " < < tubetemp < < " tube humidity " < < tubehumidlevel < < ( humidclassic ? " classic " : " systemone " ) < < " humidity " < < humidlevel ;
} else {
qWarning ( ) < < this - > sessionid < < " heated tube off " < < ( humidclassic ? " classic " : " systemone " ) < < " humidity " < < humidlevel ;
}
} else {
qWarning ( ) < < this - > sessionid < < ( humidclassic ? " classic " : " systemone " ) < < " humidity " < < humidlevel ;
}
*/
2020-01-12 00:14:01 +00:00
HumidMode humidmode = HUMID_Fixed ;
if ( tubepresent ) {
humidmode = HUMID_HeatedTube ;
} else {
if ( humidsystemone + humidclassic > 1 ) UNEXPECTED_VALUE ( humid2 , " fixed or adaptive " ) ;
if ( humidsystemone ) humidmode = HUMID_Adaptive ;
}
2019-09-29 00:44:18 +00:00
if ( add_setting ) {
2020-01-12 00:14:01 +00:00
bool humidifier_present = ( no_data = = 0 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_STATUS , humidifier_present ) ) ;
if ( humidifier_present ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_MODE , humidmode ) ) ;
if ( humidmode = = HUMID_HeatedTube ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HEATED_TUBE_TEMP , tubetemp ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , tubehumidlevel ) ) ;
} else {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , humidlevel ) ) ;
}
}
2020-01-10 15:49:08 +00:00
}
2020-01-10 21:44:00 +00:00
// Check for previously unseen data that we expect to be normal:
2020-01-10 15:49:08 +00:00
if ( this - > family = = 0 ) {
2020-01-10 21:44:00 +00:00
// F0V4
2020-01-10 15:49:08 +00:00
if ( tubetemp & & ( tubehumidlevel < 1 | | tubehumidlevel > 3 ) ) UNEXPECTED_VALUE ( tubehumidlevel , " 1-3 " ) ;
} else if ( this - > familyVersion = = 1 ) {
2020-01-10 21:44:00 +00:00
// F5V1
if ( tubepresent ) {
if ( tubetemp = = 4 ) UNEXPECTED_VALUE ( tubetemp , " != 4 " ) ;
if ( tubetemp ) {
if ( tubehumidlevel = = 0 | | tubehumidlevel > 3 ) UNEXPECTED_VALUE ( tubehumidlevel , " 1-3 " ) ;
}
}
2020-01-10 15:49:08 +00:00
} else if ( this - > familyVersion = = 2 ) {
2020-01-10 21:44:00 +00:00
// F5V2
if ( tubepresent ) {
CHECK_VALUES ( tubetemp , 0 , 3 ) ;
if ( tubetemp ) {
CHECK_VALUE ( tubehumidlevel , 1 ) ;
}
}
2020-01-10 15:49:08 +00:00
CHECK_VALUE ( humidsystemone , false ) ;
CHECK_VALUE ( humidclassic , false ) ;
2019-09-29 00:44:18 +00:00
}
2019-09-24 20:27:27 +00:00
}
bool PRS1DataChunk : : ParseSummaryF0V4 ( void )
{
2019-09-29 00:44:18 +00:00
if ( this - > family ! = 0 | | ( this - > familyVersion ! = 4 ) ) {
qWarning ( ) < < " ParseSummaryF0V4 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
2019-09-24 20:27:27 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
2019-09-29 00:44:18 +00:00
int chunk_size = this - > m_data . size ( ) ;
2019-09-30 14:23:28 +00:00
static const int minimum_sizes [ ] = { 0x18 , 7 , 7 , 0x24 , 2 , 4 , 0 , 4 , 0xb } ;
2019-09-29 00:44:18 +00:00
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
// NOTE: These are fixed sizes, but are called minimum to more closely match the F0V6 parser.
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-09-29 00:44:18 +00:00
code = data [ pos + + ] ;
// There is no hblock prior to F0V6.
size = 0 ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
size = minimum_sizes [ code ] ;
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first
2019-10-01 15:29:31 +00:00
CHECK_VALUES ( data [ pos ] & 0xF0 , 0x80 , 0xC0 ) ; // TODO: what are these?
2019-09-29 00:44:18 +00:00
if ( ( data [ pos ] & 0x0F ) ! = 1 ) { // This is the most frequent value.
2019-10-01 15:29:31 +00:00
//CHECK_VALUES(data[pos] & 0x0F, 3, 5); // TODO: what are these? 0 seems to be related to errors.
2019-09-29 00:44:18 +00:00
}
// F0V4 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
2019-10-01 15:29:31 +00:00
ok = ParseSettingsF0V4 ( data , 0x0f ) ;
CHECK_VALUE ( data [ pos + 0x11 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x12 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x13 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x14 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x15 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x16 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x17 ] , 0 ) ;
2019-09-29 00:44:18 +00:00
break ;
case 2 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
2019-10-01 15:29:31 +00:00
//CHECK_VALUES(data[pos+2], 120, 110); // probably initial pressure
//CHECK_VALUE(data[pos+3], 0); // initial IPAP on bilevel?
//CHECK_VALUES(data[pos+4], 0, 130); // minimum pressure in auto-cpap
2020-01-10 21:44:00 +00:00
this - > ParseHumidifierSetting60Series ( data [ pos + 5 ] , data [ pos + 6 ] ) ;
2019-09-29 00:44:18 +00:00
break ;
case 3 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
// F0V4 doesn't have a separate stats record like F0V6 does, the stats just follow the MaskOff data.
2019-10-01 15:29:31 +00:00
//CHECK_VALUES(data[pos+2], 130); // probably ending pressure
//CHECK_VALUE(data[pos+3], 0); // ending IPAP for bilevel? average?
//CHECK_VALUES(data[pos+4], 0, 130); // 130 pressure in auto-cpap: min pressure? 90% IPAP in bilevel?
//CHECK_VALUES(data[pos+5], 0, 130); // 130 pressure in auto-cpap, 90% EPAP in bilevel?
//CHECK_VALUE(data[pos+6], 0); // 145 maybe max pressure in Auto-CPAP?
//CHECK_VALUE(data[pos+7], 0); // Average 90% Pressure (Auto-CPAP)
//CHECK_VALUE(data[pos+8], 0); // Average CPAP (Auto-CPAP)
//CHECK_VALUES(data[pos+9], 0, 4); // or 1; PB count? LL count? minutes of something?
CHECK_VALUE ( data [ pos + 0xa ] , 0 ) ;
//CHECK_VALUE(data[pos+0xb], 0); // OA count, probably 16-bit
CHECK_VALUE ( data [ pos + 0xc ] , 0 ) ;
//CHECK_VALUE(data[pos+0xd], 0);
CHECK_VALUE ( data [ pos + 0xe ] , 0 ) ;
//CHECK_VALUE(data[pos+0xf], 0); // CA count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x10 ] , 0 ) ;
//CHECK_VALUE(data[pos+0x11], 40); // 16-bit something: 0x88, 0x26, etc. ???
//CHECK_VALUE(data[pos+0x12], 0);
//CHECK_VALUE(data[pos+0x13], 0); // 16-bit minutes in LL
//CHECK_VALUE(data[pos+0x14], 0);
//CHECK_VALUE(data[pos+0x15], 0); // minutes in PB, probably 16-bit
CHECK_VALUE ( data [ pos + 0x16 ] , 0 ) ;
//CHECK_VALUE(data[pos+0x17], 0); // 16-bit VS count
//CHECK_VALUE(data[pos+0x18], 0);
//CHECK_VALUE(data[pos+0x19], 0); // H count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x1a ] , 0 ) ;
//CHECK_VALUE(data[pos+0x1b], 0); // 0 when no PB or LL?
CHECK_VALUE ( data [ pos + 0x1c ] , 0 ) ;
//CHECK_VALUE(data[pos+0x1d], 9); // RE count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x1e ] , 0 ) ;
//CHECK_VALUE(data[pos+0x1f], 0); // FL count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x20 ] , 0 ) ;
//CHECK_VALUE(data[pos+0x21], 0x32); // 0x55, 0x19 // ???
//CHECK_VALUE(data[pos+0x22], 0x23); // 0x3f, 0x14 // Average total leak
//CHECK_VALUE(data[pos+0x23], 0x40); // 0x7d, 0x3d // ???
2019-09-29 00:44:18 +00:00
break ;
case 1 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
2019-10-01 15:29:31 +00:00
CHECK_VALUE ( data [ pos + 2 ] & ~ ( 0x40 | 8 | 4 | 2 | 1 ) , 0 ) ; // ???, seen various bit combinations
//CHECK_VALUE(data[pos+3], 0x19); // 0x17, 0x16
//CHECK_VALUES(data[pos+4], 0, 1); // or 2
//CHECK_VALUE(data[pos+5], 0x35); // 0x36, 0x36
if ( data [ pos + 6 ] ! = 1 ) { // This is the usual value.
CHECK_VALUE ( data [ pos + 6 ] & ~ ( 8 | 4 | 2 | 1 ) , 0 ) ; // On F0V23 0 seems to be related to errors, 3 seen after 90 sec large leak before turning off?
2019-09-29 00:44:18 +00:00
}
2019-10-01 15:29:31 +00:00
// pos+4 == 2, pos+6 == 10 on the session that had a time-elapsed event, maybe it shut itself off
// when approaching 24h of continuous use?
2019-09-29 00:44:18 +00:00
break ;
2019-09-30 14:23:28 +00:00
case 4 : // Time Elapsed
// For example: mask-on 5:18:49 in a session of 23:41:20 total leaves mask-off time of 18:22:31.
// That's represented by a mask-off event 19129 seconds after the mask-on, then a time-elapsed
// event after 65535 seconds, then an equipment off event after another 616 seconds.
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
// TODO: see if this event exists in other versions
break ;
2019-09-30 19:46:26 +00:00
case 5 : // Clock adjustment?
2019-09-29 00:44:18 +00:00
CHECK_VALUE ( pos , 1 ) ; // Always first
2019-09-30 14:23:28 +00:00
CHECK_VALUE ( chunk_size , 5 ) ; // and the only record in the session.
// This looks like it's minor adjustments to the clock, but 560PBT-3917 sessions 1-2 are weird:
// session 1 starts at 2015-12-23T00:01:20 and contains this event with timestamp 2015-12-23T00:05:14.
// session 2 starts at 2015-12-23T00:01:29, which suggests the event didn't change the clock.
2019-09-30 19:46:26 +00:00
//
// It looks like this happens when there are discontinuities in timestamps, for example 560P-4727:
// session 58 ends at 2015-05-26T09:53:17.
// session 59 starts at 2015-05-26T09:53:15 with an event 5 timestamp of 2015-05-26T09:53:18.
//
// So the session/chunk timestamp has gone backwards. Whenever this happens, it seems to be in
// a session with an event-5 event having a timestamp that hasn't gone backwards. So maybe
// this timestamp is the old clock before adjustment? This would explain the 560PBT-3917 sessions above.
//
// This doesn't seem particularly associated with discontinuities in the waveform data: there are
// often clock adjustments without corresponding discontinuities in the waveform, and vice versa.
// It's possible internal clock inaccuracy causes both independently.
//
// TODO: why do some machines have lots of these and others none? Maybe cellular modems make daily tweaks?
if ( false ) {
2019-09-30 14:23:28 +00:00
long value = data [ pos ] | data [ pos + 1 ] < < 8 | data [ pos + 2 ] < < 16 | data [ pos + 3 ] < < 24 ;
2019-09-30 19:46:26 +00:00
qDebug ( ) < < this - > sessionid < < " clock changing from " < < ts ( value * 1000L )
< < " to " < < ts ( this - > timestamp * 1000L )
< < " delta: " < < ( this - > timestamp - value ) ;
2019-09-30 14:23:28 +00:00
}
2019-09-29 00:44:18 +00:00
break ;
2019-10-01 15:29:31 +00:00
case 6 : // Cleared?
// Appears in the very first session when that session number is > 1.
// Presumably previous sessions were cleared out.
// TODO: add an internal event for this.
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 1 ) ; // and the only record in the session.
if ( this - > sessionid = = 1 ) UNEXPECTED_VALUE ( this - > sessionid , " >1 " ) ;
break ;
2019-09-29 00:44:18 +00:00
case 7 : // Humidifier setting change
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2020-01-10 21:44:00 +00:00
this - > ParseHumidifierSetting60Series ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-09-29 00:44:18 +00:00
break ;
2019-09-30 14:23:28 +00:00
case 8 : // CPAP-Check related, follows Mask On in CPAP-Check mode
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2020-01-28 18:04:36 +00:00
//CHECK_VALUES(data[pos+2], 0, 79); // probably 16-bit value, sometimes matches OA + H + FL + VS + RE?
2019-09-30 14:23:28 +00:00
CHECK_VALUE ( data [ pos + 3 ] , 0 ) ;
2020-01-28 18:04:36 +00:00
//CHECK_VALUES(data[pos+4], 0, 10); // probably 16-bit value
2019-09-30 14:23:28 +00:00
CHECK_VALUE ( data [ pos + 5 ] , 0 ) ;
2020-01-28 18:04:36 +00:00
//CHECK_VALUES(data[pos+6], 0, 79); // probably 16-bit value, usually the same as +2, but not always?
2019-09-30 14:23:28 +00:00
CHECK_VALUE ( data [ pos + 7 ] , 0 ) ;
2020-01-28 18:04:36 +00:00
//CHECK_VALUES(data[pos+8], 0, 10); // probably 16-bit value
2019-09-30 14:23:28 +00:00
CHECK_VALUE ( data [ pos + 9 ] , 0 ) ;
2020-01-28 18:04:36 +00:00
//CHECK_VALUES(data[pos+0xa], 0, 4); // or 0? 44 when changed pressure mid-session?
2019-09-29 00:44:18 +00:00
break ;
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
}
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-09-24 20:27:27 +00:00
2019-09-29 00:44:18 +00:00
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing bytes " ;
}
2019-09-24 20:27:27 +00:00
2019-09-29 00:44:18 +00:00
this - > duration = tt ;
2014-08-03 15:24:58 +00:00
2019-09-29 00:44:18 +00:00
return ok ;
2014-08-03 13:00:13 +00:00
}
2019-10-03 16:33:28 +00:00
// XX XX = F3V3 Humidifier bytes
2020-01-10 22:28:31 +00:00
// 43 15 = heated tube temp 5, humidity 2
// 43 14 = heated tube temp 4, humidity 2
// 63 13 = heated tube temp 3, humidity 3
// 63 11 = heated tube temp 1, humidity 3
2020-01-10 15:49:08 +00:00
// 45 08 = system one 5
2020-01-10 22:28:31 +00:00
// 44 08 = system one 4
// 43 08 = system one 3
// 42 08 = system one 2
// 41 08 = system one 1
// 40 08 = system one 0 (off)
2020-01-10 15:49:08 +00:00
// 40 60 = system one 3, no data
// 40 90 = heated tube, tube off, data=tube t=0,h=0
2020-01-10 22:28:31 +00:00
// 45 80 = classic 5
// 44 80 = classic 4
// 43 80 = classic 3
// 42 80 = classic 2
// 40 80 = classic 0 (off)
2019-10-03 16:33:28 +00:00
//
// 7 = humidity level without tube
// 8 = ? (never seen)
// 1 = ? (never seen)
// 6 = heated tube humidity level (when tube present, 0x40 all other times? including when tube is off?)
// 8 = ? (never seen)
// 7 = tube temp
// 8 = "System One" mode
// 1 = tube present
2020-01-10 15:49:08 +00:00
// 6 = no data, seems to show system one 3 in settings, only seen in session 1 briefly
// 8 = (classic mode; also seen when heated tube present but off, possibly ignored in that case)
//
2020-01-10 21:44:00 +00:00
// Note that, while containing similar fields as ParseHumidifierSetting60Series, the bit arrangement is different for F3V3!
2019-10-03 16:33:28 +00:00
void PRS1DataChunk : : ParseHumidifierSettingF3V3 ( unsigned char humid1 , unsigned char humid2 , bool add_setting )
{
if ( false ) qWarning ( ) < < this - > sessionid < < " humid " < < hex ( humid1 ) < < hex ( humid2 ) < < add_setting ;
int humidlevel = humid1 & 7 ; // Ignored when heated tube is present: humidifier setting on tube disconnect is always reported as 3
if ( humidlevel > 5 ) UNEXPECTED_VALUE ( humidlevel , " <= 5 " ) ;
CHECK_VALUE ( humid1 & 0x40 , 0x40 ) ; // seems always set, even without heated tube
CHECK_VALUE ( humid1 & 0x98 , 0 ) ; // never seen
int tubehumidlevel = ( humid1 > > 5 ) & 7 ; // This mask is a best guess based on other masks.
if ( tubehumidlevel > 5 ) UNEXPECTED_VALUE ( tubehumidlevel , " <= 5 " ) ;
CHECK_VALUE ( tubehumidlevel & 4 , 0 ) ; // never seen, but would clarify whether above mask is correct
int tubetemp = humid2 & 7 ;
if ( tubetemp > 5 ) UNEXPECTED_VALUE ( tubetemp , " <= 5 " ) ;
if ( this - > sessionid ! = 1 ) CHECK_VALUE ( humid2 & 0x60 , 0 ) ; // Only seen on 1-second session 1 of several machines, no humidifier data on chart.
2020-01-10 15:49:08 +00:00
bool humidclassic = ( humid2 & 0x80 ) ! = 0 ; // Set on classic mode reports; evidently ignored (sometimes set!) when tube is present
2019-10-03 16:33:28 +00:00
//bool no_tube? = (humid2 & 0x20) != 0; // Something tube related: whenever it is set, tube is never present (inverse is not true)
2020-01-10 15:49:08 +00:00
bool no_data = ( humid2 & 0x60 ) = = 0x60 ; // As described in chart, settings still show up
2019-10-03 16:33:28 +00:00
int tubepresent = ( humid2 & 0x10 ) ! = 0 ;
bool humidsystemone = ( humid2 & 0x08 ) ! = 0 ; // Set on "System One" humidification mode reports when tubepresent is false
2020-01-10 15:49:08 +00:00
if ( humidsystemone + tubepresent + no_data = = 0 ) CHECK_VALUE ( humidclassic , true ) ; // Always set when everything else is off in F0V4
2019-10-03 16:33:28 +00:00
if ( humidsystemone + tubepresent /*+ no_data*/ > 1 ) UNEXPECTED_VALUE ( humid2 , " one bit set " ) ; // Only one of these ever seems to be set at a time
//if (tubepresent && tubetemp == 0) CHECK_VALUE(tubehumidlevel, 0); // When the heated tube is off, tube humidity seems to be 0 in F0V4, but not F3V3
2020-01-10 15:49:08 +00:00
if ( tubepresent ) humidclassic = false ; // Classic mode bit is evidently ignored when tube is present
2019-10-03 16:33:28 +00:00
//qWarning() << this->sessionid << (humidclassic ? "C" : ".") << (humid2 & 0x20 ? "?" : ".") << (tubepresent ? "T" : ".") << (no_data ? "X" : ".") << (humidsystemone ? "1" : ".");
/*
if ( tubepresent ) {
if ( tubetemp ) {
qWarning ( ) < < this - > sessionid < < " tube temp " < < tubetemp < < " tube humidity " < < tubehumidlevel < < ( humidclassic ? " classic " : " systemone " ) < < " humidity " < < humidlevel ;
} else {
qWarning ( ) < < this - > sessionid < < " heated tube off " < < ( humidclassic ? " classic " : " systemone " ) < < " humidity " < < humidlevel ;
}
} else {
qWarning ( ) < < this - > sessionid < < ( humidclassic ? " classic " : " systemone " ) < < " humidity " < < humidlevel ;
}
*/
2020-01-12 00:14:01 +00:00
HumidMode humidmode = HUMID_Fixed ;
if ( tubepresent ) {
humidmode = HUMID_HeatedTube ;
} else {
if ( humidsystemone + humidclassic > 1 ) UNEXPECTED_VALUE ( humid2 , " fixed or adaptive " ) ;
if ( humidsystemone ) humidmode = HUMID_Adaptive ;
}
2019-10-03 16:33:28 +00:00
if ( add_setting ) {
2020-01-12 00:14:01 +00:00
bool humidifier_present = ( no_data = = 0 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_STATUS , humidifier_present ) ) ;
if ( humidifier_present ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_MODE , humidmode ) ) ;
if ( humidmode = = HUMID_HeatedTube ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HEATED_TUBE_TEMP , tubetemp ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , tubehumidlevel ) ) ;
} else {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , humidlevel ) ) ;
}
}
2019-10-03 16:33:28 +00:00
}
2020-01-10 15:49:08 +00:00
2020-01-10 22:28:31 +00:00
// Check for previously unseen data that we expect to be normal:
if ( humidclassic & & humidlevel = = 1 ) UNEXPECTED_VALUE ( humidlevel , " != 1 " ) ;
2020-01-10 15:49:08 +00:00
if ( tubepresent ) {
if ( tubetemp ) CHECK_VALUES ( tubehumidlevel , 2 , 3 ) ;
2020-01-10 22:28:31 +00:00
if ( tubetemp = = 2 ) UNEXPECTED_VALUE ( tubetemp , " != 2 " ) ;
2020-01-10 15:49:08 +00:00
}
2019-10-03 16:33:28 +00:00
}
2019-10-03 00:21:43 +00:00
// Support for 1061T, 1160P
// logic largely borrowed from ParseSettingsF3V6, values based on sample data
bool PRS1DataChunk : : ParseSettingsF3V3 ( const unsigned char * data , int /*size*/ )
2019-07-24 00:54:39 +00:00
{
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2019-10-03 00:21:43 +00:00
FlexMode flexmode = FLEX_Unknown ;
switch ( data [ 2 ] ) {
2019-10-03 16:33:28 +00:00
case 0 : cpapmode = PRS1_MODE_CPAP ; break ; // "CPAP" mode
case 1 : cpapmode = PRS1_MODE_S ; break ; // "S" mode
2019-10-03 00:21:43 +00:00
case 2 : cpapmode = PRS1_MODE_ST ; break ; // "S/T" mode; pressure seems variable?
2019-10-03 16:33:28 +00:00
case 4 : cpapmode = PRS1_MODE_PC ; break ; // "PC" mode? Usually "PC - AVAPS", see setting 1 below
2019-10-03 00:21:43 +00:00
default :
UNEXPECTED_VALUE ( data [ 2 ] , " known device mode " ) ;
break ;
}
2019-08-29 01:30:25 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
2019-10-03 00:21:43 +00:00
switch ( data [ 3 ] ) {
case 0 : // 0 = None
flexmode = FLEX_None ;
2019-10-03 16:33:28 +00:00
if ( cpapmode ! = PRS1_MODE_CPAP ) {
CHECK_VALUES ( cpapmode , PRS1_MODE_S , PRS1_MODE_ST ) ;
}
2019-10-03 00:21:43 +00:00
break ;
case 1 : // 1 = Bi-Flex, only seen with "S - Bi-Flex"
flexmode = FLEX_BiFlex ;
CHECK_VALUE ( cpapmode , PRS1_MODE_S ) ;
break ;
case 2 : // 2 = AVAPS: usually "PC - AVAPS", sometimes "S/T - AVAPS"
flexmode = FLEX_AVAPS ;
2019-10-03 16:33:28 +00:00
CHECK_VALUES ( cpapmode , PRS1_MODE_ST , PRS1_MODE_PC ) ;
2019-10-03 00:21:43 +00:00
break ;
default :
UNEXPECTED_VALUE ( data [ 3 ] , " known flex mode " ) ;
break ;
}
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_MODE , ( int ) flexmode ) ) ;
int epap = data [ 4 ] + ( data [ 5 ] < < 1 ) ; // 0x82 = EPAP 13 cmH2O; 0x78 = EPAP 12 cmH2O; 0x50 = EPAP 8 cmH2O
int min_ipap = data [ 6 ] + ( data [ 7 ] < < 1 ) ; // 0xA0 = IPAP 16 cmH2O; 0xBE = 19 cmH2O min IPAP (in AVAPS); 0x78 = IPAP 12 cmH2O
int max_ipap = data [ 8 ] + ( data [ 9 ] < < 1 ) ; // 0xAA = ???; 0x12C = 30 cmH2O max IPAP (in AVAPS); 0x78 = ???
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP , epap ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MIN , min_ipap ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , max_ipap ) ) ;
// TODO: calculte PS or min/max PS? Create IPAP event when not AVAPS?
2019-10-03 16:33:28 +00:00
if ( flexmode = = FLEX_None | | flexmode = = FLEX_AVAPS ) {
2020-03-23 00:00:09 +00:00
int rise_time = data [ 0xa ] ; // 1 = Rise Time Setting 1, 2 = Rise Time Setting 2, 3 = Rise Time Setting 3
if ( cpapmode = = PRS1_MODE_CPAP ) {
CHECK_VALUE ( rise_time , 0 ) ;
} else {
if ( rise_time < 1 | | rise_time > 6 ) UNEXPECTED_VALUE ( rise_time , " 1-6 " ) ; // TODO: what is 0?
CHECK_VALUES ( data [ 0xb ] , 0 , 1 ) ; // 1 = Rise Time Lock (in "None" and AVAPS flex mode)
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RISE_TIME , rise_time ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RISE_TIME_LOCK , data [ 0xb ] = = 1 ) ) ;
}
2019-10-03 16:33:28 +00:00
} else {
2020-03-23 00:00:09 +00:00
CHECK_VALUES ( data [ 0xa ] , 0 , 3 ) ; // TODO: May also be Bi-Flex 3? But how is this different from [0xc] below?
2019-10-03 16:33:28 +00:00
CHECK_VALUE ( data [ 0xb ] , 0 ) ;
}
if ( flexmode = = FLEX_BiFlex ) {
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LEVEL , data [ 0xc ] ) ) ; // 3 = Bi-Flex 3 (in bi-flex mode)
CHECK_VALUE ( data [ 0xc ] , 3 ) ;
2019-10-03 16:33:28 +00:00
CHECK_VALUE ( data [ 0x0a ] , data [ 0xc ] ) ;
} else {
CHECK_VALUE ( data [ 0xc ] , 0 ) ;
}
2019-10-03 00:21:43 +00:00
CHECK_VALUE ( data [ 0xd ] , 0 ) ;
2020-03-23 00:00:09 +00:00
if ( flexmode ! = FLEX_AVAPS & & cpapmode ! = PRS1_MODE_CPAP ) CHECK_VALUE ( data [ 0xe ] , 0x14 ) ; // 0x14 = ??? in S and non-AVAPS S/T and PC
2019-10-03 16:33:28 +00:00
if ( cpapmode = = PRS1_MODE_CPAP ) CHECK_VALUE ( data [ 0xe ] , 0 ) ;
2019-10-03 00:21:43 +00:00
if ( flexmode = = FLEX_AVAPS ) {
2020-03-23 00:00:09 +00:00
if ( data [ 0xe ] < 24 | | data [ 0xe ] > 65 ) UNEXPECTED_VALUE ( data [ 0xe ] , " 24-65 " ) ;
2019-10-03 00:21:43 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TIDAL_VOLUME , data [ 0xe ] * 10.0 ) ) ;
}
2020-03-23 00:00:09 +00:00
int breath_rate = data [ 0xf ] ;
int timed_inspiration = data [ 0x10 ] ;
switch ( cpapmode ) {
case PRS1_MODE_CPAP :
case PRS1_MODE_S :
CHECK_VALUE ( breath_rate , 0 ) ;
CHECK_VALUE ( timed_inspiration , 0 ) ;
break ;
case PRS1_MODE_PC :
CHECK_VALUE ( flexmode , FLEX_AVAPS ) ;
2020-03-23 15:45:10 +00:00
CHECK_VALUE ( breath_rate , 0 ) ; // only ever seen 0 on reports so far
2020-03-23 00:00:09 +00:00
CHECK_VALUE ( timed_inspiration , 30 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Fixed ) ) ;
2020-03-23 15:45:10 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_RATE , breath_rate ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ScaledSettingEvent ( PRS1_SETTING_BACKUP_TIMED_INSPIRATION , timed_inspiration , 0.1 ) ) ;
break ;
case PRS1_MODE_ST :
if ( flexmode = = FLEX_AVAPS ) {
2020-03-23 15:45:10 +00:00
if ( breath_rate ) { // can be 0 on reports
2020-03-23 03:19:35 +00:00
CHECK_VALUES ( breath_rate , 9 , 10 ) ;
}
if ( timed_inspiration < 10 | | timed_inspiration > 30 ) UNEXPECTED_VALUE ( timed_inspiration , " 10-30 " ) ;
2020-03-23 15:45:10 +00:00
} else if ( flexmode = = FLEX_None ) {
CHECK_VALUES ( breath_rate , 0x0C , 0x0A ) ; // 0xC = Breath Rate 12, 0xA = Breath Rate 10, can this be 0?
CHECK_VALUES ( timed_inspiration , 10 , 20 ) ; // 0xA = Timed Inspiration 1, 0x14 = Time Inspiration 2
2020-03-23 00:00:09 +00:00
}
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Fixed ) ) ;
2020-03-23 15:45:10 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_RATE , breath_rate ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ScaledSettingEvent ( PRS1_SETTING_BACKUP_TIMED_INSPIRATION , timed_inspiration , 0.1 ) ) ;
2020-03-23 15:45:10 +00:00
break ;
default :
UNEXPECTED_VALUE ( cpapmode , " CPAP, S, S/T, or PC " ) ;
break ;
2020-03-23 00:00:09 +00:00
}
2019-10-03 00:21:43 +00:00
CHECK_VALUE ( data [ 0x11 ] , 0 ) ;
//CHECK_VALUE(data[0x12], 0x1E, 0x0F); // 0x1E = ramp time 30 minutes, 0x0F = ramp time 15 minutes
//CHECK_VALUE(data[0x13], 0x3C, 0x5A, 0x28); // 0x3C = ramp pressure 6 cmH2O, 0x28 = ramp pressure 4 cmH2O, 0x5A = ramp pressure 9 cmH2O
CHECK_VALUE ( data [ 0x14 ] , 0 ) ; // the ramp pressure is probably a 16-bit value like the ones above are
int ramp_time = data [ 0x12 ] ;
int ramp_pressure = data [ 0x13 ] ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , ramp_time ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , ramp_pressure ) ) ;
2019-10-03 16:33:28 +00:00
this - > ParseHumidifierSettingF3V3 ( data [ 0x15 ] , data [ 0x16 ] , true ) ;
2019-10-03 00:21:43 +00:00
// Menu options?
2019-10-03 16:33:28 +00:00
CHECK_VALUES ( data [ 0x17 ] , 0x10 , 0x90 ) ; // 0x10 = resist 1; 0x90 = resist 1, resist lock
2019-10-03 00:21:43 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , ( data [ 0x17 ] & 0x80 ) ! = 0 ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , 1 ) ) ; // only value seen so far, CHECK_VALUES above will flag any others
2019-10-03 16:33:28 +00:00
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TUBING_LOCK , ( data [ 0x18 ] & 0x80 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HOSE_DIAMETER , ( data [ 0x18 ] & 0x7f ) ) ) ;
2019-10-03 16:33:28 +00:00
CHECK_VALUES ( data [ 0x18 ] & 0x7f , 22 , 15 ) ; // 0x16 = tubing 22; 0x0F = tubing 15, 0x96 = tubing 22 with lock
2019-10-03 00:21:43 +00:00
// Alarms?
CHECK_VALUE ( data [ 0x19 ] , 0 ) ;
CHECK_VALUE ( data [ 0x1a ] , 0 ) ;
CHECK_VALUE ( data [ 0x1b ] , 0 ) ;
2019-10-01 15:38:16 +00:00
return true ;
}
2019-10-03 00:21:43 +00:00
// borrowed largely from ParseSummaryF5V012
2019-10-01 15:38:16 +00:00
bool PRS1DataChunk : : ParseSummaryF3V3 ( void )
{
2019-10-03 00:21:43 +00:00
if ( this - > family ! = 3 | | ( this - > familyVersion > 3 ) ) {
qWarning ( ) < < " ParseSummaryF3V3 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
2019-10-01 15:38:16 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
2019-10-03 00:21:43 +00:00
int chunk_size = this - > m_data . size ( ) ;
QVector < int > minimum_sizes = { 0x1b , 3 , 5 , 9 } ;
// NOTE: These are fixed sizes, but are called minimum to more closely match the F0V6 parser.
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-10-03 00:21:43 +00:00
code = data [ pos + + ] ;
// There is no hblock prior to F3V6.
size = 0 ;
if ( code < minimum_sizes . length ( ) ) {
// make sure the handlers below don't go past the end of the buffer
size = minimum_sizes [ code ] ;
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
2019-10-09 14:24:29 +00:00
// NOTE: F3V3 doesn't use 16-bit time deltas in its summary events, it uses absolute timestamps!
// It's possible that these are 24-bit, but haven't yet seen a timestamp that large.
2019-10-03 00:21:43 +00:00
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( data [ pos ] , 0 ) ;
/*
CHECK_VALUE ( data [ pos ] & 0xF0 , 0 ) ; // TODO: what are these?
if ( ( data [ pos ] & 0x0F ) ! = 1 ) { // This is the most frequent value.
//CHECK_VALUES(data[pos] & 0x0F, 3, 5); // TODO: what are these? 0 seems to be related to errors.
}
*/
// F3V3 doesn't have a separate settings record like F3V6 does, the settings just follow the EquipmentOn data.
ok = this - > ParseSettingsF3V3 ( data , size ) ;
break ;
case 2 : // Mask On
2019-10-09 14:24:29 +00:00
tt = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2019-10-03 00:21:43 +00:00
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
2019-10-09 14:24:29 +00:00
CHECK_VALUE ( data [ pos + 2 ] , 0 ) ; // may be high byte of timestamp
2019-10-03 16:33:28 +00:00
this - > ParseHumidifierSettingF3V3 ( data [ pos + 3 ] , data [ pos + 4 ] ) ;
2019-10-03 00:21:43 +00:00
break ;
case 3 : // Mask Off
2019-10-09 14:24:29 +00:00
tt = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2019-10-03 00:21:43 +00:00
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
// F3V3 doesn't have a separate stats record like F3V6 does, the stats just follow the MaskOff data.
2019-10-09 14:24:29 +00:00
CHECK_VALUE ( data [ pos + 0x2 ] , 0 ) ; // may be high byte of timestamp
2019-10-03 16:33:28 +00:00
CHECK_VALUE ( data [ pos + 0x3 ] , 0 ) ; // probably OA count, but the only sample data is missing .002 files, so we can't yet verify
2019-10-03 00:21:43 +00:00
CHECK_VALUE ( data [ pos + 0x4 ] , 0 ) ;
2019-10-03 16:33:28 +00:00
//CHECK_VALUE(data[pos+0x5], 0); // CA count, probably 16-bit
2019-10-03 00:21:43 +00:00
CHECK_VALUE ( data [ pos + 0x6 ] , 0 ) ;
2019-10-03 16:33:28 +00:00
//CHECK_VALUE(data[pos+0x7], 0); // H count, probably 16-bit
2019-10-03 00:21:43 +00:00
CHECK_VALUE ( data [ pos + 0x8 ] , 0 ) ;
break ;
case 1 : // Equipment Off
2019-10-09 14:24:29 +00:00
tt = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
2019-10-03 00:21:43 +00:00
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
2019-10-09 14:24:29 +00:00
CHECK_VALUE ( data [ pos + 2 ] , 0 ) ; // may be high byte of timestamp
2019-10-03 00:21:43 +00:00
break ;
/*
case 5 : // Clock adjustment? See ParseSummaryF0V4.
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 5 ) ; // and the only record in the session.
if ( false ) {
long value = data [ pos ] | data [ pos + 1 ] < < 8 | data [ pos + 2 ] < < 16 | data [ pos + 3 ] < < 24 ;
qDebug ( ) < < this - > sessionid < < " clock changing from " < < ts ( value * 1000L )
< < " to " < < ts ( this - > timestamp * 1000L )
< < " delta: " < < ( this - > timestamp - value ) ;
}
break ;
case 6 : // Cleared?
// Appears in the very first session when that session number is > 1.
// Presumably previous sessions were cleared out.
// TODO: add an internal event for this.
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 1 ) ; // and the only record in the session.
if ( this - > sessionid = = 1 ) UNEXPECTED_VALUE ( this - > sessionid , " >1 " ) ;
break ;
case 7 : // ???
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
break ;
case 8 : // ???
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // Since 7 and 8 seem to occur near each other, let's assume 8 also has a timestamp
CHECK_VALUE ( pos , 1 ) ;
CHECK_VALUE ( chunk_size , 3 ) ;
CHECK_VALUE ( data [ pos ] , 0 ) ; // and alert us if the timestamp is nonzero
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ;
break ;
case 9 : // Humidifier setting change
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2020-01-10 21:44:00 +00:00
this - > ParseHumidifierSetting60Series ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-10-03 00:21:43 +00:00
break ;
*/
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
}
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-10-03 00:21:43 +00:00
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing bytes " ;
}
this - > duration = tt ;
return ok ;
2019-07-24 00:54:39 +00:00
}
2019-07-24 20:50:51 +00:00
// Originally based on ParseSummaryF5V3, with changes observed in ventilator sample data
//
// TODO: surely there will be a way to merge ParseSummary (FV3) loops and abstract the machine-specific
// encodings into another function or class, but that's probably worth pursuing only after
// the details have been figured out.
bool PRS1DataChunk : : ParseSummaryF3V6 ( void )
{
if ( this - > family ! = 3 | | this - > familyVersion ! = 6 ) {
qWarning ( ) < < " ParseSummaryF3V6 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2020-01-23 00:52:49 +00:00
static const int minimum_sizes [ ] = { 1 , 0x25 , 9 , 7 , 4 , 2 , 1 , 2 , 2 , 1 , 0x18 , 2 , 4 } ; // F5V3 = { 1, 0x38, 4, 2, 4, 0x1e, 2, 4, 9 };
2019-07-24 20:50:51 +00:00
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
// NOTE: The sizes contained in hblock can vary, even within a single machine, as can the length of hblock itself!
// TODO: hardcoding this is ugly, think of a better approach
if ( chunk_size < minimum_sizes [ 0 ] + minimum_sizes [ 1 ] + minimum_sizes [ 2 ] ) {
qWarning ( ) < < this - > sessionid < < " summary data too short: " < < chunk_size ;
return false ;
}
2020-03-09 17:57:43 +00:00
// We've once seen a short summary with no mask-on/off: just equipment-on, settings, 2, equipment-off
// (And we've seen something similar in F5V3.)
if ( chunk_size < 58 ) UNEXPECTED_VALUE ( chunk_size , " >= 58 " ) ;
2019-07-24 20:50:51 +00:00
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-07-24 20:50:51 +00:00
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
if ( size < minimum_sizes [ code ] ) {
2020-01-23 00:34:42 +00:00
UNEXPECTED_VALUE ( size , minimum_sizes [ code ] ) ;
2019-07-24 20:50:51 +00:00
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " too small " < < size < < " < " < < minimum_sizes [ code ] ;
2020-01-23 00:34:42 +00:00
if ( code ! = 1 ) { // Settings are variable-length, so shorter settings slices aren't fatal.
ok = false ;
break ;
}
2019-07-24 20:50:51 +00:00
}
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first?
//CHECK_VALUE(data[pos], 0x10); // usually 0x10 for 1030X, sometimes 0x40 or 0x80 are set in addition or instead
CHECK_VALUE ( size , 1 ) ;
break ;
case 1 : // Settings
ok = this - > ParseSettingsF3V6 ( data + pos , size ) ;
break ;
2019-09-20 16:59:14 +00:00
case 2 : // seems equivalent to F5V3 #9, comes right after settings, usually 9 bytes, identical values
2019-09-19 14:33:23 +00:00
// TODO: This may be structurally similar to settings: a list of (code, length, value).
2019-07-24 20:50:51 +00:00
CHECK_VALUE ( data [ pos ] , 0 ) ;
CHECK_VALUE ( data [ pos + 1 ] , 1 ) ;
2019-09-19 14:33:23 +00:00
//CHECK_VALUE(data[pos+2], 0); // Apnea Alarm (0=off, 1=10, 2=20)
2019-09-20 16:59:14 +00:00
if ( data [ pos + 2 ] ! = 0 ) {
CHECK_VALUES ( data [ pos + 2 ] , 1 , 2 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_APNEA_ALARM , data [ pos + 2 ] * 10 ) ) ;
}
2019-07-24 20:50:51 +00:00
CHECK_VALUE ( data [ pos + 3 ] , 1 ) ;
CHECK_VALUE ( data [ pos + 4 ] , 1 ) ;
CHECK_VALUE ( data [ pos + 5 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 6 ] , 2 ) ;
CHECK_VALUE ( data [ pos + 7 ] , 1 ) ;
2019-09-20 16:59:14 +00:00
CHECK_VALUE ( data [ pos + 8 ] , 0 ) ; // 1 = patient disconnect alarm of 15 sec on F5V3, not sure where time is encoded
if ( size > 9 ) {
CHECK_VALUE ( data [ pos + 9 ] , 3 ) ;
CHECK_VALUE ( data [ pos + 10 ] , 1 ) ;
CHECK_VALUE ( data [ pos + 11 ] , 0 ) ;
CHECK_VALUE ( size , 12 ) ;
}
2019-07-24 20:50:51 +00:00
break ;
case 4 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
break ;
case 5 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
break ;
2020-01-23 00:52:49 +00:00
case 6 : // Ventilator CPAP stats, presumably per mask-on slice
//CHECK_VALUE(data[pos], 0x3C); // Average CPAP
break ;
2019-07-24 20:50:51 +00:00
case 7 : // Ventilator EPAP stats, presumably per mask-on slice
//CHECK_VALUE(data[pos], 0x69); // Average EPAP
//CHECK_VALUE(data[pos+1], 0x80); // Average 90% EPAP
break ;
case 8 : // Ventilator IPAP stats, presumably per mask-on slice
//CHECK_VALUE(data[pos], 0x86); // Average IPAP
//CHECK_VALUE(data[pos+1], 0xA8); // Average 90% IPAP
break ;
case 0xa : // Patient statistics, presumably per mask-on slice
//CHECK_VALUE(data[pos], 0x00); // 16-bit OA count
CHECK_VALUE ( data [ pos + 1 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+2], 0x00); // 16-bit CA count
CHECK_VALUE ( data [ pos + 3 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+4], 0x00); // 16-bit minutes in LL
CHECK_VALUE ( data [ pos + 5 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+6], 0x0A); // 16-bit VS count
2019-07-27 02:53:15 +00:00
//CHECK_VALUE(data[pos+7], 0x00); // We've actually seen someone with more than 255 VS in a night!
2019-07-24 20:50:51 +00:00
//CHECK_VALUE(data[pos+8], 0x01); // 16-bit H count (partial)
CHECK_VALUE ( data [ pos + 9 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+0xa], 0x00); // 16-bit H count (partial)
CHECK_VALUE ( data [ pos + 0xb ] , 0x00 ) ;
//CHECK_VALUE(data[pos+0xc], 0x00); // 16-bit RE count
CHECK_VALUE ( data [ pos + 0xd ] , 0x00 ) ;
//CHECK_VALUE(data[pos+0xe], 0x3e); // average total leak
//CHECK_VALUE(data[pos+0xf], 0x03); // 16-bit H count (partial)
CHECK_VALUE ( data [ pos + 0x10 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+0x11], 0x11); // average breath rate
//CHECK_VALUE(data[pos+0x12], 0x41); // average TV / 10
//CHECK_VALUE(data[pos+0x13], 0x60); // average % PTB
//CHECK_VALUE(data[pos+0x14], 0x0b); // average minute vent
//CHECK_VALUE(data[pos+0x15], 0x1d); // average leak? (similar position to F5V3, similar delta to total leak)
//CHECK_VALUE(data[pos+0x16], 0x00); // 16-bit minutes in PB
CHECK_VALUE ( data [ pos + 0x17 ] , 0x00 ) ;
break ;
case 3 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
//CHECK_VALUES(data[pos+2], 1, 4); // bitmask, have seen 1, 4, 6, 0x41
//CHECK_VALUE(data[pos+3], 0x17); // 0x16, etc.
2019-09-19 14:33:23 +00:00
//CHECK_VALUES(data[pos+4], 0, 1); // or 2
2019-07-24 20:50:51 +00:00
//CHECK_VALUE(data[pos+5], 0x15); // 0x16, etc.
//CHECK_VALUES(data[pos+6], 0, 1); // or 2
break ;
case 0xc : // Humidier setting change
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
break ;
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
break ;
}
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-07-24 20:50:51 +00:00
this - > duration = tt ;
return ok ;
}
// Based initially on ParseSettingsF5V3. Many of the codes look the same, like always starting with 0, 0x35 looking like
// a humidifier setting, etc., but the contents are sometimes a bit different, such as mode values and pressure settings.
//
// new settings to find: ...
bool PRS1DataChunk : : ParseSettingsF3V6 ( const unsigned char * data , int size )
{
static const QMap < int , int > expected_lengths = { { 0x1e , 3 } , { 0x35 , 2 } } ;
bool ok = true ;
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2019-09-20 16:59:14 +00:00
FlexMode flexmode = FLEX_Unknown ;
2019-07-24 20:50:51 +00:00
// F5V3 and F3V6 use a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
static const float GAIN = 0.125 ; // TODO: parameterize this somewhere better
2020-01-23 00:52:49 +00:00
int fixed_pressure = 0 ;
2019-07-24 20:50:51 +00:00
int fixed_epap = 0 ;
int fixed_ipap = 0 ;
int min_ipap = 0 ;
int max_ipap = 0 ;
2020-03-23 00:00:09 +00:00
int breath_rate ;
int timed_inspiration ;
2019-07-24 20:50:51 +00:00
// Parse the nested data structure which contains settings
int pos = 0 ;
do {
int code = data [ pos + + ] ;
int len = data [ pos + + ] ;
int expected_len = 1 ;
if ( expected_lengths . contains ( code ) ) {
expected_len = expected_lengths [ code ] ;
}
//CHECK_VALUE(len, expected_len);
if ( len < expected_len ) {
qWarning ( ) < < this - > sessionid < < " setting " < < code < < " too small " < < len < < " < " < < expected_len ;
ok = false ;
break ;
}
if ( pos + len > size ) {
qWarning ( ) < < this - > sessionid < < " setting " < < code < < " @ " < < pos < < " longer than remaining slice " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 : // Device Mode
CHECK_VALUE ( pos , 2 ) ; // always first?
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
switch ( data [ pos ] ) {
2020-01-23 00:52:49 +00:00
case 0 : cpapmode = PRS1_MODE_CPAP ; break ; // "CPAP" mode
2019-08-29 01:30:25 +00:00
case 1 : cpapmode = PRS1_MODE_S ; break ; // "S" mode
case 2 : cpapmode = PRS1_MODE_ST ; break ; // "S/T" mode; pressure seems variable?
case 4 : cpapmode = PRS1_MODE_PC ; break ; // "PC" mode? Usually "PC - AVAPS", see setting 1 below
2019-07-24 20:50:51 +00:00
default :
UNEXPECTED_VALUE ( data [ pos ] , " known device mode " ) ;
break ;
}
2019-08-29 01:52:59 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
2019-07-24 20:50:51 +00:00
break ;
2019-09-20 16:59:14 +00:00
case 1 : // Flex Mode
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 16:59:14 +00:00
switch ( data [ pos ] ) {
2019-09-23 16:39:20 +00:00
case 0 : // 0 = None
flexmode = FLEX_None ;
2020-01-23 00:52:49 +00:00
if ( cpapmode ! = PRS1_MODE_CPAP ) {
CHECK_VALUES ( cpapmode , PRS1_MODE_S , PRS1_MODE_ST ) ;
}
2019-09-23 16:39:20 +00:00
break ;
case 1 : // 1 = Bi-Flex, only seen with "S - Bi-Flex"
flexmode = FLEX_BiFlex ;
CHECK_VALUE ( cpapmode , PRS1_MODE_S ) ;
break ;
case 2 : // 2 = AVAPS: usually "PC - AVAPS", sometimes "S/T - AVAPS"
flexmode = FLEX_AVAPS ;
CHECK_VALUES ( cpapmode , PRS1_MODE_ST , PRS1_MODE_PC ) ;
break ;
default :
UNEXPECTED_VALUE ( data [ pos ] , " known flex mode " ) ;
break ;
2019-09-20 16:59:14 +00:00
}
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_MODE , ( int ) flexmode ) ) ;
2019-07-24 20:50:51 +00:00
break ;
2019-09-20 16:59:14 +00:00
case 2 : // ??? Maybe AAM?
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
CHECK_VALUE ( data [ pos ] , 0 ) ;
break ;
2020-01-23 00:52:49 +00:00
case 3 : // CPAP Pressure
CHECK_VALUE ( len , 1 ) ;
fixed_pressure = data [ pos ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE , fixed_pressure , GAIN ) ) ;
break ;
2019-07-24 20:50:51 +00:00
case 4 : // EPAP Pressure
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
// pressures seem variable on practice, maybe due to ramp or leaks?
fixed_epap = data [ pos ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP , fixed_epap , GAIN ) ) ;
break ;
case 7 : // IPAP Pressure
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
// pressures seem variable on practice, maybe due to ramp or leaks?
fixed_ipap = data [ pos ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP , fixed_ipap , GAIN ) ) ;
2019-08-20 16:03:51 +00:00
// TODO: We need to revisit whether PS should be shown as a setting.
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS , fixed_ipap - fixed_epap , GAIN ) ) ;
if ( fixed_epap = = 0 ) UNEXPECTED_VALUE ( fixed_epap , " >0 " ) ;
2019-07-24 20:50:51 +00:00
break ;
case 8 : // Min IPAP
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-08-20 16:03:51 +00:00
CHECK_VALUE ( fixed_ipap , 0 ) ;
2019-07-24 20:50:51 +00:00
min_ipap = data [ pos ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MIN , min_ipap , GAIN ) ) ;
2019-08-20 16:03:51 +00:00
// TODO: We need to revisit whether PS should be shown as a setting.
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MIN , min_ipap - fixed_epap , GAIN ) ) ;
if ( fixed_epap = = 0 ) UNEXPECTED_VALUE ( fixed_epap , " >0 " ) ;
2019-07-24 20:50:51 +00:00
break ;
case 9 : // Max IPAP
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-08-20 16:03:51 +00:00
CHECK_VALUE ( fixed_ipap , 0 ) ;
if ( min_ipap = = 0 ) UNEXPECTED_VALUE ( min_ipap , " >0 " ) ;
2019-07-24 20:50:51 +00:00
max_ipap = data [ pos ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , max_ipap , GAIN ) ) ;
2019-08-20 16:03:51 +00:00
// TODO: We need to revisit whether PS should be shown as a setting.
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MAX , max_ipap - fixed_epap , GAIN ) ) ;
if ( fixed_epap = = 0 ) UNEXPECTED_VALUE ( fixed_epap , " >0 " ) ;
2019-07-24 20:50:51 +00:00
break ;
case 0x19 : // Tidal Volume (AVAPS)
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-08-29 01:52:59 +00:00
CHECK_VALUES ( cpapmode , PRS1_MODE_ST , PRS1_MODE_PC ) ;
2019-09-20 16:59:14 +00:00
CHECK_VALUE ( flexmode , FLEX_AVAPS ) ;
2019-07-27 02:53:15 +00:00
//CHECK_VALUE(data[pos], 47); // gain 10.0
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TIDAL_VOLUME , data [ pos ] * 10.0 ) ) ;
2019-07-24 20:50:51 +00:00
break ;
2019-09-23 17:43:28 +00:00
case 0x1e : // (Backup) Breath Rate (S/T and PC)
CHECK_VALUE ( len , 3 ) ;
2019-08-29 01:52:59 +00:00
CHECK_VALUES ( cpapmode , PRS1_MODE_ST , PRS1_MODE_PC ) ;
2019-09-23 17:43:28 +00:00
switch ( data [ pos ] ) {
case 0 : // Breath Rate Off
// TODO: Is this mode essentially bilevel? The pressure graphs are confusing.
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Off ) ) ;
2020-03-23 00:00:09 +00:00
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 2 ] , 0 ) ;
2019-09-23 17:43:28 +00:00
break ;
//case 1: // Breath Rate Auto in F5V3 setting 0x14
case 2 : // Breath Rate (fixed BPM)
2020-03-23 00:00:09 +00:00
breath_rate = data [ pos + 1 ] ;
timed_inspiration = data [ pos + 2 ] ;
2020-03-23 03:19:35 +00:00
if ( breath_rate < 9 | | breath_rate > 12 ) UNEXPECTED_VALUE ( breath_rate , " 9-12 " ) ;
2020-03-23 00:00:09 +00:00
if ( timed_inspiration < 10 | | timed_inspiration > 20 ) UNEXPECTED_VALUE ( timed_inspiration , " 10-20 " ) ;
2019-09-23 17:43:28 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Fixed ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_RATE , breath_rate ) ) ;
this - > AddEvent ( new PRS1ScaledSettingEvent ( PRS1_SETTING_BACKUP_TIMED_INSPIRATION , timed_inspiration , 0.1 ) ) ;
2019-09-23 17:43:28 +00:00
break ;
default :
2020-03-23 00:00:09 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 2 ) ; // 0 = Breath Rate off (S), 2 = fixed BPM (1 = auto on F5V3 setting 0x14, haven't seen it on F3V6 yet)
2019-09-23 17:43:28 +00:00
break ;
}
2019-07-24 20:50:51 +00:00
break ;
2020-03-23 00:00:09 +00:00
//0x2b: Ramp type sounds like it's linear for F3V6 unless AAM is enabled, so no setting may be needed.
2019-07-24 20:50:51 +00:00
case 0x2c : // Ramp Time
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
if ( data [ pos ] ! = 0 ) { // 0 == ramp off, and ramp pressure setting doesn't appear
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , data [ pos ] ) ) ;
}
break ;
2019-09-19 14:33:23 +00:00
case 0x2d : // Ramp Pressure (with ASV/ventilator pressure encoding), only present when ramp is on
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , data [ pos ] , GAIN ) ) ;
break ;
case 0x2e : // Bi-Flex level or Rise Time
// On F5V3 the first byte could specify Bi-Flex or Rise Time, and second byte contained the value.
2019-09-23 16:39:20 +00:00
// On F3V6 there's only one byte, which seems to correspond to Rise Time on the reports with flex
// mode None or AVAPS and to Bi-Flex Setting (level) in Bi-Flex mode.
CHECK_VALUE ( len , 1 ) ;
if ( flexmode = = FLEX_BiFlex ) {
// Bi-Flex level
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LEVEL , data [ pos ] ) ) ;
} else if ( flexmode = = FLEX_None | | flexmode = = FLEX_AVAPS ) {
// Rise time
2019-11-13 16:39:02 +00:00
if ( data [ pos ] < 1 | | data [ pos ] > 6 ) UNEXPECTED_VALUE ( data [ pos ] , " 1-6 " ) ; // 1-6 have been seen
2019-09-23 16:39:20 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RISE_TIME , data [ pos ] ) ) ;
}
2020-03-23 00:00:09 +00:00
// Timed inspiration specified in the backup breath rate.
2019-07-24 20:50:51 +00:00
break ;
case 0x2f : // Rise Time lock? (was flex lock on F0V6, 0x80 for locked)
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-19 14:33:23 +00:00
if ( cpapmode = = PRS1_MODE_S ) {
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ; // Bi-Flex Lock
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LOCK , data [ pos ] ! = 0 ) ) ;
2019-09-19 14:33:23 +00:00
} else {
CHECK_VALUE ( data [ pos ] , 0 ) ; // Rise Time Lock? not yet observed on F3V6
2020-03-23 00:00:09 +00:00
//this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RISE_TIME_LOCK, data[pos] != 0));
2019-09-19 14:33:23 +00:00
}
2019-07-24 20:50:51 +00:00
break ;
case 0x35 : // Humidifier setting
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 2 ) ;
2019-07-24 20:50:51 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos ] , data [ pos + 1 ] , true ) ;
break ;
case 0x36 : // Mask Resistance Lock
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
CHECK_VALUE ( data [ pos ] , 0 ) ; // 0x80 = locked on F5V3, not yet observed on F3V6
2019-09-20 04:15:40 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , data [ pos ] ! = 0 ) ) ;
2019-07-24 20:50:51 +00:00
break ;
case 0x38 : // Mask Resistance
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
if ( data [ pos ] ! = 0 ) { // 0 == mask resistance off
2020-03-23 03:19:35 +00:00
if ( data [ pos ] < 1 | | data [ pos ] > 5 ) UNEXPECTED_VALUE ( data [ pos ] , " 1-5 " ) ;
2019-09-20 04:15:40 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , data [ pos ] ) ) ;
2019-07-24 20:50:51 +00:00
}
break ;
2020-01-23 00:52:49 +00:00
case 0x39 : // Tubing Type Lock
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2020-01-23 00:52:49 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TUBING_LOCK , data [ pos ] ! = 0 ) ) ;
2019-07-24 20:50:51 +00:00
break ;
case 0x3b : // Tubing Type
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
if ( data [ pos ] ! = 0 ) {
CHECK_VALUES ( data [ pos ] , 2 , 1 ) ; // 15HT = 2, 15 = 1, 22 = 0, though report only says "15" for 15HT
}
2019-09-20 01:29:33 +00:00
this - > ParseTubingTypeV3 ( data [ pos ] ) ;
2019-07-24 20:50:51 +00:00
break ;
case 0x3c : // View Optional Screens
2019-09-23 17:43:28 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-07-24 20:50:51 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
2019-09-20 01:29:33 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_SHOW_AHI , data [ pos ] ! = 0 ) ) ;
2019-07-24 20:50:51 +00:00
break ;
default :
2020-03-23 00:00:09 +00:00
UNEXPECTED_VALUE ( code , " known setting " ) ;
2019-07-24 20:50:51 +00:00
qDebug ( ) < < " Unknown setting: " < < hex < < code < < " in " < < this - > sessionid < < " at " < < pos ;
this - > AddEvent ( new PRS1UnknownDataEvent ( QByteArray ( ( const char * ) data , size ) , pos , len ) ) ;
break ;
}
pos + = len ;
} while ( ok & & pos + 2 < = size ) ;
2019-08-20 16:48:15 +00:00
2019-07-24 20:50:51 +00:00
return ok ;
}
2019-10-01 15:38:16 +00:00
bool PRS1DataChunk : : ParseSettingsF5V012 ( const unsigned char * data , int /*size*/ )
2019-05-25 23:21:46 +00:00
{
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2014-08-03 15:24:58 +00:00
2019-10-02 15:55:44 +00:00
float GAIN = PRS1PressureSettingEvent : : GAIN ;
if ( this - > familyVersion = = 2 ) GAIN = 0.125f ; // TODO: parameterize this somewhere better
2019-10-01 22:30:32 +00:00
int imax_pressure = data [ 0x2 ] ;
2014-08-06 07:08:34 +00:00
int imin_epap = data [ 0x3 ] ;
int imax_epap = data [ 0x4 ] ;
int imin_ps = data [ 0x5 ] ;
int imax_ps = data [ 0x6 ] ;
2014-08-03 15:24:58 +00:00
2019-10-01 22:30:32 +00:00
// Only one mode available, so apparently there's no byte in the settings that encodes it?
2019-08-29 01:30:25 +00:00
cpapmode = PRS1_MODE_ASV ;
2019-05-25 23:21:46 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MIN , imin_epap ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MAX , imax_epap ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MIN , imin_epap + imin_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , imax_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MIN , imin_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MAX , imax_ps ) ) ;
2019-05-26 21:36:12 +00:00
2019-10-02 15:55:44 +00:00
//CHECK_VALUE(data[0x07], 1, 2); // 1 = backup breath rate "Auto"; 2 = fixed BPM, see below
//CHECK_VALUE(data[0x08], 0); // backup "Breath Rate" in mode 2
//CHECK_VALUE(data[0x09], 0); // backup "Timed Inspiration" (gain 0.1) in mode 2
2020-03-23 00:00:09 +00:00
int pos = 0x7 ;
int backup_mode = data [ pos ] ;
int breath_rate ;
int timed_inspiration ;
switch ( backup_mode ) {
case 0 :
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Off ) ) ;
break ;
case 1 :
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Auto ) ) ;
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 2 ] , 0 ) ;
break ;
case 2 :
breath_rate = data [ pos + 1 ] ;
timed_inspiration = data [ pos + 2 ] ;
2020-03-23 03:19:35 +00:00
if ( breath_rate < 4 | | breath_rate > 29 ) UNEXPECTED_VALUE ( breath_rate , " 4-29 " ) ;
if ( timed_inspiration < 5 | | timed_inspiration > 20 ) UNEXPECTED_VALUE ( timed_inspiration , " 5-20 " ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Fixed ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_RATE , breath_rate ) ) ;
this - > AddEvent ( new PRS1ScaledSettingEvent ( PRS1_SETTING_BACKUP_TIMED_INSPIRATION , timed_inspiration , 0.1 ) ) ;
break ;
default :
UNEXPECTED_VALUE ( backup_mode , " 0-2 " ) ;
break ;
}
2019-05-26 21:36:12 +00:00
2019-10-01 22:30:32 +00:00
int ramp_time = data [ 0x0a ] ;
2019-10-02 15:55:44 +00:00
int ramp_pressure = data [ 0x0b ] ;
2019-05-26 21:36:12 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , ramp_time ) ) ;
2019-10-02 15:55:44 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , ramp_pressure , GAIN ) ) ;
2019-05-26 21:36:12 +00:00
2020-01-12 21:53:43 +00:00
quint8 flex = data [ 0x0c ] ;
this - > ParseFlexSettingF5V012 ( flex , cpapmode ) ;
2019-10-01 22:30:32 +00:00
if ( this - > familyVersion = = 0 ) { // TODO: either split this into two functions or use size to differentiate like FV3 parsers do
2019-10-02 15:55:44 +00:00
// TODO: Is there another flag for F5V0? Reports say "Bypass System One Humidification" as an option?
2020-01-10 17:57:18 +00:00
this - > ParseHumidifierSetting50Series ( data [ 0x0d ] , true ) ;
2019-10-01 22:30:32 +00:00
pos = 0xe ;
} else {
2020-01-10 21:44:00 +00:00
// 60-Series machines have a 2-byte humidfier setting.
this - > ParseHumidifierSetting60Series ( data [ 0x0d ] , data [ 0x0e ] , true ) ;
2019-10-01 22:30:32 +00:00
pos = 0xf ;
}
2019-10-02 15:55:44 +00:00
// TODO: may differ between F5V0 and F5V12
// 0x01, 0x41 = auto-on, view AHI, tubing type = 15
// 0x41, 0x41 = auto-on, view AHI, tubing type = 15, resist lock
// 0x42, 0x01 = (no auto-on), view AHI, tubing type = 22, resist lock, tubing lock
2019-10-02 22:57:42 +00:00
// 0x00, 0x41 = auto-on, view AHI, tubing type = 22, no tubing lock
// 0x0B, 0x41 = mask resist 1, tube lock, tubing type = 15, auto-on, view AHI
// 0x09, 0x01 = mask resist 1, tubing 15, view AHI
// 0x19, 0x41 = mask resist 3, tubing 15, auto-on, view AHI
// 0x29, 0x41 = mask resist 5, tubing 15, auto-on, view AHI
2019-10-02 15:55:44 +00:00
// 1 = view AHI
// 4 = auto-on
2019-10-02 22:57:42 +00:00
// 1 = tubing type: 0=22, 1=15
// 2 = tubing lock
// 38 = mask resist level
2019-10-02 15:55:44 +00:00
// 4 = resist lock
2019-10-02 22:57:42 +00:00
int resist_level = ( data [ pos ] > > 3 ) & 7 ; // 0x09 resist=1, 0x11 resist=2, 0x19=resist 3, 0x29=resist 5
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , ( data [ pos ] & 0x40 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , resist_level ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HOSE_DIAMETER , ( data [ pos ] & 0x01 ) ? 15 : 22 ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TUBING_LOCK , ( data [ pos ] & 0x02 ) ! = 0 ) ) ;
2019-10-02 22:57:42 +00:00
CHECK_VALUE ( data [ pos ] & ( 0x80 | 0x04 ) , 0 ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_ON , ( data [ pos + 1 ] & 0x40 ) ! = 0 ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_SHOW_AHI , ( data [ pos + 1 ] & 1 ) ! = 0 ) ) ;
2019-10-01 22:30:32 +00:00
CHECK_VALUE ( data [ pos + 1 ] & ~ ( 0x40 | 1 ) , 0 ) ;
2019-10-02 15:55:44 +00:00
2019-12-01 23:01:32 +00:00
CHECK_VALUES ( data [ pos + 2 ] , 0 , 1 ) ; // 1 = apnea alarm 10
2019-10-02 15:55:44 +00:00
CHECK_VALUE ( data [ pos + 3 ] , 0 ) ; // low MV alarm?
2019-12-01 23:01:32 +00:00
if ( data [ pos + 4 ] ) {
CHECK_VALUES ( data [ pos + 4 ] , 1 , 2 ) ; // 1 = disconnect alarm 15, 2 = disconnect alarm 60
}
2014-08-03 15:24:58 +00:00
2019-10-01 15:38:16 +00:00
return true ;
}
2019-10-01 22:30:32 +00:00
// borrowed largely from ParseSummaryF0V4
2019-10-01 15:38:16 +00:00
bool PRS1DataChunk : : ParseSummaryF5V012 ( void )
{
2019-10-01 22:30:32 +00:00
if ( this - > family ! = 5 | | ( this - > familyVersion > 2 ) ) {
qWarning ( ) < < " ParseSummaryF5V012 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
2019-10-01 15:38:16 +00:00
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
2019-10-01 22:30:32 +00:00
int chunk_size = this - > m_data . size ( ) ;
QVector < int > minimum_sizes ;
switch ( this - > familyVersion ) {
case 0 : minimum_sizes = { 0x12 , 4 , 3 , 0x1f } ; break ;
2019-10-02 15:55:44 +00:00
case 1 : minimum_sizes = { 0x13 , 7 , 5 , 0x20 , 0 , 4 , 0 , 2 , 2 , 4 } ; break ;
2019-10-02 22:57:42 +00:00
case 2 : minimum_sizes = { 0x13 , 7 , 5 , 0x22 , 0 , 4 , 0 , 2 , 2 , 4 } ; break ;
2019-10-01 22:30:32 +00:00
}
// NOTE: These are fixed sizes, but are called minimum to more closely match the F0V6 parser.
2019-10-01 15:38:16 +00:00
2019-10-01 22:30:32 +00:00
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-10-01 22:30:32 +00:00
code = data [ pos + + ] ;
// There is no hblock prior to F0V6.
size = 0 ;
if ( code < minimum_sizes . length ( ) ) {
// make sure the handlers below don't go past the end of the buffer
size = minimum_sizes [ code ] ;
2020-01-10 21:44:00 +00:00
} else {
// We can't defer warning until later, because F5V0 doesn't have slice 4-9.
UNEXPECTED_VALUE ( code , " known slice code " ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
}
2019-10-01 22:30:32 +00:00
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first
/*
CHECK_VALUE ( data [ pos ] & 0xF0 , 0 ) ; // TODO: what are these?
if ( ( data [ pos ] & 0x0F ) ! = 1 ) { // This is the most frequent value.
//CHECK_VALUES(data[pos] & 0x0F, 3, 5); // TODO: what are these? 0 seems to be related to errors.
}
*/
2019-10-03 00:21:43 +00:00
// F5V012 doesn't have a separate settings record like F5V3 does, the settings just follow the EquipmentOn data.
2019-10-01 22:30:32 +00:00
ok = this - > ParseSettingsF5V012 ( data , size ) ;
/*
CHECK_VALUE ( data [ pos + 0x11 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x12 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x13 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x14 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x15 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x16 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 0x17 ] , 0 ) ;
*/
break ;
case 2 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
/*
//CHECK_VALUES(data[pos+2], 120, 110); // probably initial pressure
//CHECK_VALUE(data[pos+3], 0); // initial IPAP on bilevel?
//CHECK_VALUES(data[pos+4], 0, 130); // minimum pressure in auto-cpap
2020-01-10 21:44:00 +00:00
this - > ParseHumidifierSetting60Series ( data [ pos + 5 ] , data [ pos + 6 ] ) ;
2019-10-01 22:30:32 +00:00
*/
break ;
case 3 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
2019-10-03 00:21:43 +00:00
// F5V012 doesn't have a separate stats record like F5V3 does, the stats just follow the MaskOff data.
2019-10-01 22:30:32 +00:00
/*
//CHECK_VALUES(data[pos+2], 130); // probably ending pressure
//CHECK_VALUE(data[pos+3], 0); // ending IPAP for bilevel? average?
//CHECK_VALUES(data[pos+4], 0, 130); // 130 pressure in auto-cpap: min pressure? 90% IPAP in bilevel?
//CHECK_VALUES(data[pos+5], 0, 130); // 130 pressure in auto-cpap, 90% EPAP in bilevel?
//CHECK_VALUE(data[pos+6], 0); // 145 maybe max pressure in Auto-CPAP?
//CHECK_VALUE(data[pos+7], 0); // Average 90% Pressure (Auto-CPAP)
//CHECK_VALUE(data[pos+8], 0); // Average CPAP (Auto-CPAP)
//CHECK_VALUES(data[pos+9], 0, 4); // or 1; PB count? LL count? minutes of something?
CHECK_VALUE ( data [ pos + 0xa ] , 0 ) ;
//CHECK_VALUE(data[pos+0xb], 0); // OA count, probably 16-bit
CHECK_VALUE ( data [ pos + 0xc ] , 0 ) ;
//CHECK_VALUE(data[pos+0xd], 0);
CHECK_VALUE ( data [ pos + 0xe ] , 0 ) ;
//CHECK_VALUE(data[pos+0xf], 0); // CA count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x10 ] , 0 ) ;
//CHECK_VALUE(data[pos+0x11], 40); // 16-bit something: 0x88, 0x26, etc. ???
//CHECK_VALUE(data[pos+0x12], 0);
//CHECK_VALUE(data[pos+0x13], 0); // 16-bit minutes in LL
//CHECK_VALUE(data[pos+0x14], 0);
//CHECK_VALUE(data[pos+0x15], 0); // minutes in PB, probably 16-bit
CHECK_VALUE ( data [ pos + 0x16 ] , 0 ) ;
//CHECK_VALUE(data[pos+0x17], 0); // 16-bit VS count
//CHECK_VALUE(data[pos+0x18], 0);
//CHECK_VALUE(data[pos+0x19], 0); // H count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x1a ] , 0 ) ;
//CHECK_VALUE(data[pos+0x1b], 0); // 0 when no PB or LL?
CHECK_VALUE ( data [ pos + 0x1c ] , 0 ) ;
//CHECK_VALUE(data[pos+0x1d], 9); // RE count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x1e ] , 0 ) ;
//CHECK_VALUE(data[pos+0x1f], 0); // FL count, probably 16-bit
CHECK_VALUE ( data [ pos + 0x20 ] , 0 ) ;
//CHECK_VALUE(data[pos+0x21], 0x32); // 0x55, 0x19 // ???
//CHECK_VALUE(data[pos+0x22], 0x23); // 0x3f, 0x14 // Average total leak
//CHECK_VALUE(data[pos+0x23], 0x40); // 0x7d, 0x3d // ???
*/
break ;
case 1 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
2020-01-10 17:57:18 +00:00
if ( this - > familyVersion = = 0 ) {
2020-01-12 00:23:43 +00:00
//CHECK_VALUE(data[pos+2], 1); // Usually 1, also seen 0, 6, and 7.
2020-01-10 17:57:18 +00:00
ParseHumidifierSetting50Series ( data [ pos + 3 ] ) ;
}
/* Possibly F5V12?
2019-10-01 22:30:32 +00:00
CHECK_VALUE ( data [ pos + 2 ] & ~ ( 0x40 | 8 | 4 | 2 | 1 ) , 0 ) ; // ???, seen various bit combinations
//CHECK_VALUE(data[pos+3], 0x19); // 0x17, 0x16
//CHECK_VALUES(data[pos+4], 0, 1); // or 2
//CHECK_VALUE(data[pos+5], 0x35); // 0x36, 0x36
if ( data [ pos + 6 ] ! = 1 ) { // This is the usual value.
CHECK_VALUE ( data [ pos + 6 ] & ~ ( 8 | 4 | 2 | 1 ) , 0 ) ; // On F0V23 0 seems to be related to errors, 3 seen after 90 sec large leak before turning off?
}
// pos+4 == 2, pos+6 == 10 on the session that had a time-elapsed event, maybe it shut itself off
// when approaching 24h of continuous use?
*/
break ;
2019-10-02 15:55:44 +00:00
case 5 : // Clock adjustment? See ParseSummaryF0V4.
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 5 ) ; // and the only record in the session.
if ( false ) {
long value = data [ pos ] | data [ pos + 1 ] < < 8 | data [ pos + 2 ] < < 16 | data [ pos + 3 ] < < 24 ;
qDebug ( ) < < this - > sessionid < < " clock changing from " < < ts ( value * 1000L )
< < " to " < < ts ( this - > timestamp * 1000L )
< < " delta: " < < ( this - > timestamp - value ) ;
}
break ;
case 6 : // Cleared?
// Appears in the very first session when that session number is > 1.
// Presumably previous sessions were cleared out.
// TODO: add an internal event for this.
CHECK_VALUE ( pos , 1 ) ; // Always first
CHECK_VALUE ( chunk_size , 1 ) ; // and the only record in the session.
if ( this - > sessionid = = 1 ) UNEXPECTED_VALUE ( this - > sessionid , " >1 " ) ;
break ;
2019-10-02 22:57:42 +00:00
case 7 : // ???
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2019-10-02 15:55:44 +00:00
break ;
2019-10-02 22:57:42 +00:00
case 8 : // ???
2019-12-01 23:16:57 +00:00
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This also adds to the total duration (otherwise it won't match report)
2019-10-02 15:55:44 +00:00
break ;
2019-10-01 22:30:32 +00:00
case 9 : // Humidifier setting change
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2020-01-10 21:44:00 +00:00
this - > ParseHumidifierSetting60Series ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-10-01 22:30:32 +00:00
break ;
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
ok = false ; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
break ;
}
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-05-26 21:36:12 +00:00
2019-10-01 22:30:32 +00:00
if ( ok & & pos ! = chunk_size ) {
qWarning ( ) < < this - > sessionid < < ( this - > size ( ) - pos ) < < " trailing bytes " ;
}
this - > duration = tt ;
return ok ;
2019-05-26 21:36:12 +00:00
}
2020-01-10 15:49:08 +00:00
// Flex F0V2 confirmed
// 0x00 = None
2020-01-12 21:53:43 +00:00
// 0x81 = C-Flex 1, lock off (AutoCPAP mode)
// 0x82 = Bi-Flex 2 (Bi-Level mode)
// 0x89 = A-Flex 1 (AutoCPAP mode)
// 0x8A = A-Flex 2, lock off (AutoCPAP mode)
// 0x8B = C-Flex+ 3, lock off (CPAP mode)
// 0x93 = Rise Time 3 (AutoBiLevel mode)
2020-01-10 15:49:08 +00:00
// Flex F0V4 confirmed
2020-01-12 21:53:43 +00:00
// 0x00 = None
// 0x81 = Bi-Flex 1 (AutoBiLevel mode)
// 0x81 = C-Flex 1 (AutoCPAP mode)
// 0x82 = C-Flex 2 (CPAP mode)
// 0x82 = C-Flex 2 (CPAP-Check mode)
2020-01-12 22:27:12 +00:00
// 0x82 = C-Flex 2 (Auto-Trial mode)
2020-01-12 21:53:43 +00:00
// 0x83 = Bi-Flex 3 (Bi-Level mode)
// 0x89 = A-Flex 1 (AutoCPAP mode)
// 0x8A = C-Flex+ 2 (CPAP mode)
// 0x8A = C-Flex+ 2, lock off (CPAP-Check mode)
// 0x8A = A-Flex 2, lock off (Auto-Trial mode)
2020-01-28 18:04:36 +00:00
// 0xCB = C-Flex+ 3 (CPAP-Check mode), C-Flex+ Lock on
2020-01-12 21:53:43 +00:00
//
// 0x8A = A-Flex 1 (AutoCPAP mode)
// 0x8B = C-Flex+ 3 (CPAP mode)
// 0x8B = A-Flex 3 (AutoCPAP mode)
// 8 = enabled
// 1 = rise time
// 8 = C-Flex+ / A-Flex (depending on mode)
// 3 = level
void PRS1DataChunk : : ParseFlexSettingF0V234 ( quint8 flex , int cpapmode )
2019-05-26 21:36:12 +00:00
{
2020-01-12 21:53:43 +00:00
FlexMode flexmode = FLEX_None ;
bool enabled = ( flex & 0x80 ) ! = 0 ;
2020-01-28 18:04:36 +00:00
bool lock = ( flex & 0x40 ) ! = 0 ;
2020-01-12 21:53:43 +00:00
bool risetime = ( flex & 0x10 ) ! = 0 ;
bool plusmode = ( flex & 0x08 ) ! = 0 ;
2014-08-03 15:24:58 +00:00
int flexlevel = flex & 0x03 ;
2020-01-28 18:04:36 +00:00
if ( flex & ( 0x20 | 0x04 ) ) UNEXPECTED_VALUE ( flex , " known bits " ) ;
if ( this - > familyVersion = = 2 ) {
CHECK_VALUE ( lock , false ) ; // haven't observed this yet
}
2020-01-12 21:53:43 +00:00
if ( enabled ) {
if ( flexlevel < 1 ) UNEXPECTED_VALUE ( flexlevel , " != 0 " ) ;
if ( risetime ) {
flexmode = FLEX_RiseTime ;
CHECK_VALUE ( cpapmode , PRS1_MODE_AUTOBILEVEL ) ;
CHECK_VALUE ( plusmode , 0 ) ;
} else if ( plusmode ) {
switch ( cpapmode ) {
case PRS1_MODE_CPAP :
case PRS1_MODE_CPAPCHECK :
flexmode = FLEX_CFlexPlus ;
break ;
case PRS1_MODE_AUTOCPAP :
case PRS1_MODE_AUTOTRIAL :
flexmode = FLEX_AFlex ;
break ;
default :
2020-01-12 22:27:12 +00:00
HEX ( flex ) ;
2020-01-12 21:53:43 +00:00
UNEXPECTED_VALUE ( cpapmode , " expected C-Flex+/A-Flex mode " ) ;
break ;
}
} else {
switch ( cpapmode ) {
case PRS1_MODE_CPAP :
case PRS1_MODE_CPAPCHECK :
case PRS1_MODE_AUTOCPAP :
2020-01-12 22:27:12 +00:00
case PRS1_MODE_AUTOTRIAL :
2020-01-12 21:53:43 +00:00
flexmode = FLEX_CFlex ;
break ;
case PRS1_MODE_BILEVEL :
case PRS1_MODE_AUTOBILEVEL :
flexmode = FLEX_BiFlex ;
break ;
default :
2020-01-12 22:27:12 +00:00
HEX ( flex ) ;
2020-01-12 21:53:43 +00:00
UNEXPECTED_VALUE ( cpapmode , " expected mode " ) ;
break ;
}
}
}
2014-08-03 15:24:58 +00:00
2019-05-25 23:21:46 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_MODE , ( int ) flexmode ) ) ;
2020-01-12 22:27:12 +00:00
if ( flexmode ! = FLEX_None ) {
2020-01-12 21:53:43 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LEVEL , flexlevel ) ) ;
}
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LOCK , lock ) ) ;
2020-01-12 21:53:43 +00:00
}
// Flex F5V0 confirmed
// 0x81 = Bi-Flex 1 (ASV mode)
// 0x82 = Bi-Flex 2 (ASV mode)
// 0x83 = Bi-Flex 3 (ASV mode)
// Flex F5V1 confirmed
2020-01-13 00:34:51 +00:00
// 0x81 = Bi-Flex 1 (ASV mode)
2020-01-12 21:53:43 +00:00
// 0x82 = Bi-Flex 2 (ASV mode)
// 0x83 = Bi-Flex 3 (ASV mode)
// 0xC9 = Rise Time 1, Rise Time Lock (ASV mode)
2020-01-13 00:34:51 +00:00
// 0x8A = Rise Time 2 (ASV mode) (Shows "ASV - None" in mode summary, but then rise time in details)
// 0x8B = Rise Time 3 (ASV mode) (breath rate auto)
// 0x08 = Rise Time 2 (ASV mode) (falls back to level=2? bits encode level=0)
2020-01-12 21:53:43 +00:00
// Flex F5V2 confirmed
// 0x02 = Bi-Flex 2 (ASV mode) (breath rate auto, but min/max PS=0)
2020-01-13 00:34:51 +00:00
// this could be different from F5V01, or PS=0 could disable flex?
2020-01-12 21:53:43 +00:00
// 8 = ? (once was 0 when rise time was on and backup breathing was off, rise time level was also 0 in that case)
// (was also 0 on F5V2)
// 4 = Rise Time Lock
// 8 = Rise Time (vs. Bi-Flex)
// 3 = level
void PRS1DataChunk : : ParseFlexSettingF5V012 ( quint8 flex , int cpapmode )
{
2020-01-13 00:34:51 +00:00
FlexMode flexmode = FLEX_Unknown ;
2020-01-12 21:53:43 +00:00
bool unknown = ( flex & 0x80 ) ! = 0 ;
bool lock = ( flex & 0x40 ) ! = 0 ;
bool risetime = ( flex & 0x08 ) ! = 0 ;
int flexlevel = flex & 0x03 ;
if ( flex & ( 0x20 | 0x10 | 0x04 ) ) UNEXPECTED_VALUE ( flex , " known bits " ) ;
CHECK_VALUE ( cpapmode , PRS1_MODE_ASV ) ;
if ( this - > familyVersion = = 0 ) {
2020-01-13 00:34:51 +00:00
CHECK_VALUE ( unknown , true ) ;
CHECK_VALUE ( lock , false ) ;
CHECK_VALUE ( risetime , false ) ;
if ( flexlevel = = 0 ) UNEXPECTED_VALUE ( flexlevel , " 1-3 " ) ;
2020-01-12 21:53:43 +00:00
} else if ( this - > familyVersion = = 1 ) {
2020-01-13 00:34:51 +00:00
if ( unknown = = false ) {
CHECK_VALUE ( flex , 0x08 ) ;
2020-03-23 00:00:09 +00:00
flexlevel = 2 ; // TODO: Why do reports say Rise Time 2 for this value?
2020-01-12 21:53:43 +00:00
}
2020-01-13 00:34:51 +00:00
if ( lock ) CHECK_VALUE ( risetime , true ) ; // so far we've only seen rise time lock, but this could mean bi-flex lock as well
if ( flexlevel = = 0 & & unknown ) UNEXPECTED_VALUE ( flexlevel , " 1-3 " ) ;
2020-01-12 21:53:43 +00:00
} else {
2020-01-13 00:34:51 +00:00
CHECK_VALUE ( flex , 0x02 ) ; // only seen one example, unsure if it matches F5V01
}
2020-03-23 00:00:09 +00:00
// We're only confident of values where the high bit is set
2020-01-13 00:34:51 +00:00
if ( unknown ) {
if ( risetime ) {
flexmode = FLEX_RiseTime ;
} else {
flexmode = FLEX_BiFlex ;
2020-01-12 21:53:43 +00:00
}
}
2020-01-13 00:34:51 +00:00
// TODO: rise or bi-flex lock once we're confident about it
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_MODE , ( int ) flexmode ) ) ;
if ( flexmode ! = FLEX_Unknown ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LEVEL , flexlevel ) ) ;
}
2014-08-17 12:56:05 +00:00
}
2014-08-03 15:24:58 +00:00
2019-05-26 21:36:12 +00:00
2020-01-10 15:49:08 +00:00
// Humid F0V2 confirmed
2020-01-10 17:39:54 +00:00
// 0x00 = Off (presumably no humidifier present)
2020-01-10 15:49:08 +00:00
// 0x80 = Off
2020-01-10 17:39:54 +00:00
// 0x81 = 1
2020-01-10 15:49:08 +00:00
// 0x82 = 2
// 0x83 = 3
// 0x84 = 4
// 0x85 = 5
2020-01-10 17:39:54 +00:00
// Humid F5V0 confirmed
// 0x00 = Off (presumably no humidifier present)
// 0x80 = Off
// 0x81 = 1, bypass = no
2020-01-10 15:49:08 +00:00
// 0x82 = 2, bypass = no
2020-01-12 00:23:43 +00:00
// 0x83 = 3, bypass = no
// 0x84 = 4, bypass = no
// 0x85 = 5, bypass = no
2020-01-10 15:49:08 +00:00
2020-01-10 17:57:18 +00:00
void PRS1DataChunk : : ParseHumidifierSetting50Series ( int humid , bool add_setting )
2019-05-26 22:46:26 +00:00
{
2020-01-10 15:49:08 +00:00
if ( humid & ( 0x40 | 0x20 | 0x10 | 0x08 ) ) UNEXPECTED_VALUE ( humid , " known bits " ) ;
2019-06-04 00:12:17 +00:00
2020-01-10 17:57:18 +00:00
bool humidifier_present = ( ( humid & 0x80 ) ! = 0 ) ; // humidifier connected
int humidlevel = humid & 7 ; // humidification level
2020-01-10 15:49:08 +00:00
2020-01-12 00:14:01 +00:00
HumidMode humidmode = HUMID_Fixed ; // 50-Series didn't have adaptive or heated tube humidification
2020-01-10 17:57:18 +00:00
if ( add_setting ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_STATUS , humidifier_present ) ) ;
2020-01-12 00:14:01 +00:00
if ( humidifier_present ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_MODE , humidmode ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , humidlevel ) ) ;
}
2020-01-10 17:57:18 +00:00
}
2019-06-04 00:12:17 +00:00
2020-01-10 17:39:54 +00:00
// Check for truly unexpected values:
2019-06-04 00:12:17 +00:00
if ( humidlevel > 5 ) UNEXPECTED_VALUE ( humidlevel , " <= 5 " ) ;
2020-01-10 17:39:54 +00:00
if ( ! humidifier_present ) CHECK_VALUE ( humidlevel , 0 ) ;
2019-05-26 22:46:26 +00:00
}
2019-06-07 00:19:46 +00:00
// The below is based on fixing the fileVersion == 3 parsing in ParseSummary() based
// on our understanding of slices from F0V23. The switch values come from sample files.
2019-06-06 20:08:40 +00:00
bool PRS1DataChunk : : ParseComplianceF0V6 ( void )
{
if ( this - > family ! = 0 | | this - > familyVersion ! = 6 ) {
2019-06-08 02:05:52 +00:00
qWarning ( ) < < " ParseComplianceF0V6 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
2019-06-06 20:08:40 +00:00
return false ;
}
2019-06-07 00:19:46 +00:00
// TODO: hardcoding this is ugly, think of a better approach
if ( this - > m_data . size ( ) < 82 ) {
qWarning ( ) < < this - > sessionid < < " compliance data too short: " < < this - > m_data . size ( ) ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
static const int expected_sizes [ ] = { 1 , 0x34 , 9 , 4 , 2 , 2 , 4 , 8 } ;
static const int ncodes = sizeof ( expected_sizes ) / sizeof ( int ) ;
for ( int i = 0 ; i < ncodes ; i + + ) {
if ( this - > hblock . contains ( i ) ) {
CHECK_VALUE ( this - > hblock [ i ] , expected_sizes [ i ] ) ;
} else {
UNEXPECTED_VALUE ( this - > hblock . contains ( i ) , true ) ;
}
}
bool ok = true ;
int pos = 0 ;
int code , size ;
2019-06-07 18:32:53 +00:00
int tt = 0 ;
2019-06-07 00:19:46 +00:00
do {
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
if ( size < expected_sizes [ code ] ) {
2020-01-23 00:34:42 +00:00
UNEXPECTED_VALUE ( size , expected_sizes [ code ] ) ;
2019-06-07 00:19:46 +00:00
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " too small " < < size < < " < " < < expected_sizes [ code ] ;
2020-01-23 00:34:42 +00:00
if ( code ! = 1 ) { // Settings are variable-length, so shorter settings slices aren't fatal.
ok = false ;
break ;
}
2019-06-07 00:19:46 +00:00
}
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 :
// always first? Maybe equipmenton? Maybe 0 was always equipmenton, even in F0V23?
2019-06-07 18:32:53 +00:00
CHECK_VALUE ( pos , 1 ) ;
2019-06-07 20:03:20 +00:00
//CHECK_VALUES(data[pos], 1, 3); // sometimes 7?
2019-06-07 00:19:46 +00:00
break ;
2019-06-07 18:32:53 +00:00
case 1 : // Settings
2019-06-07 00:19:46 +00:00
// This is where ParseSummaryF0V6 started (after "3 bytes that don't follow the pattern")
// Both compliance and summary files seem to have the same length for this slice, so maybe the
// settings are the same?
ok = this - > ParseSettingsF0V6 ( data + pos , size ) ;
break ;
2019-06-07 18:32:53 +00:00
case 3 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
2019-06-14 00:31:21 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-07 18:32:53 +00:00
case 4 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
2019-06-07 00:19:46 +00:00
break ;
case 7 :
2019-06-07 18:32:53 +00:00
// Always follows mask off?
2019-06-07 20:03:20 +00:00
//CHECK_VALUES(data[pos], 0x01, 0x00); // sometimes 32, 4
2019-06-07 00:19:46 +00:00
CHECK_VALUE ( data [ pos + 1 ] , 0x00 ) ;
2019-06-07 20:03:20 +00:00
//CHECK_VALUES(data[pos+2], 0x00, 0x01); // sometimes 11, 3, 15
2019-06-07 00:19:46 +00:00
CHECK_VALUE ( data [ pos + 3 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+4], 0x05, 0x0A); // 00
CHECK_VALUE ( data [ pos + 5 ] , 0x00 ) ;
//CHECK_VALUE(data[pos+6], 0x64, 0x69); // 6E, 6D, 6E, 6E, 80
//CHECK_VALUE(data[pos+7], 0x3d, 0x5c); // 6A, 6A, 6B, 6C, 80
break ;
2019-06-07 18:32:53 +00:00
case 2 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
2019-06-07 00:19:46 +00:00
//CHECK_VALUE(data[pos+2], 0x08); // 0x01
//CHECK_VALUE(data[pos+3], 0x14); // 0x12
//CHECK_VALUE(data[pos+4], 0x01); // 0x00
//CHECK_VALUE(data[pos+5], 0x22); // 0x28
2019-06-07 20:03:20 +00:00
//CHECK_VALUE(data[pos+6], 0x02); // sometimes 1, 0
2019-06-07 00:19:46 +00:00
CHECK_VALUE ( data [ pos + 7 ] , 0x00 ) ; // 0x00
CHECK_VALUE ( data [ pos + 8 ] , 0x00 ) ; // 0x00
break ;
2019-06-07 20:03:20 +00:00
case 6 : // Humidier setting change
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2019-06-14 00:31:21 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-06-07 18:32:53 +00:00
break ;
2019-06-07 00:19:46 +00:00
default :
2019-06-07 18:32:53 +00:00
UNEXPECTED_VALUE ( code , " known slice code " ) ;
2019-06-07 00:19:46 +00:00
break ;
}
pos + = size ;
} while ( ok & & pos < chunk_size ) ;
2019-06-07 18:32:53 +00:00
this - > duration = tt ;
2019-06-07 00:19:46 +00:00
return ok ;
}
2019-09-20 01:29:33 +00:00
void PRS1DataChunk : : ParseTubingTypeV3 ( unsigned char type )
{
int diam ;
switch ( type ) {
case 0 : diam = 22 ; break ;
case 1 : diam = 15 ; break ;
case 2 : diam = 15 ; break ; // 15HT, though the reports only say "15" for DreamStation models
case 3 : diam = 12 ; break ; // seen on DreamStation Go models
default :
UNEXPECTED_VALUE ( type , " known tubing type " ) ;
return ;
}
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HOSE_DIAMETER , diam ) ) ;
}
2020-01-10 15:49:08 +00:00
// F0V6 confirmed
// 90 B0 = HT=3!,H=3!,data=none [no humidifier appears to ignore HT and H bits and show HT=3,H=3 in details]
// 8C 6C = HT=3, H=3, data=none
// 80 00 = nothing listed in details, data=none, only seen on 400G and 502G
// 54 B4 = HT=5, H=5, data=tube
// 50 90 = HT=4, H=4, data=tube
// 4C 6C = HT=3, H=3, data=tube
2020-01-11 20:13:24 +00:00
// 48 68 = HT=3, H=2, data=tube
// 40 60 = HT=3, H=Off, data=tube t=3,h=0
2020-01-10 15:49:08 +00:00
// 50 50 = HT=2, H=4, data=tube
// 4C 4C = HT=2, H=3, data=tube
2020-01-11 20:13:24 +00:00
// 50 30 = HT=1, H=4, data=tube
2020-01-10 15:49:08 +00:00
// 4C 0C = HT=off, H=3, data=tube t=0,h=3
2020-01-11 20:13:24 +00:00
// 34 74 = HT=3, H=5, data=adaptive (5)
2020-01-10 15:49:08 +00:00
// 50 B0 = HT=5, H=4, adaptive
// 30 B0 = HT=3, H=4, data=adaptive (4)
// 30 50 = HT=3, H=4, data=adaptive (4)
// 30 10 = HT=3!,H=4, data=adaptive (4) [adaptive mode appears to ignore HT bits and show HT=3 in details]
// 30 70 = HT=3, H=4, data=adaptive (4)
// 2C 6C = HT=3, H=3, data=adaptive (3)
// 28 08 = H=2, data=adaptive (2), no details (400G)
// 28 48 = HT=3!,H=2, data=adaptive (2) [adaptive mode appears to ignore HT bits and show HT=3 in details]
// 28 68 = HT=3, H=2, data=adaptive (2)
2020-01-11 20:13:24 +00:00
// 24 64 = HT=3, H=1, data=adaptive (1)
2020-01-10 15:49:08 +00:00
// 20 60 = HT=3, H=off, data=adaptive (0)
2020-01-11 20:13:24 +00:00
// 14 74 = HT=3, H=5, data=fixed (5)
// 10 70 = HT=3, H=4, data=fixed (4)
2020-01-10 15:49:08 +00:00
// 0C 6C = HT=3, H=3, data=fixed (3)
2020-01-11 20:13:24 +00:00
// 08 48 = HT=3, H=2, data=fixed (2)
// 08 68 = HT=3, H=2, data=fixed (2)
2020-01-10 15:49:08 +00:00
// 04 64 = HT=3, H=1, data=fixed (1)
2020-01-11 20:13:24 +00:00
// 00 00 = HT=3, H=off, data=fixed (0)
2020-01-10 15:49:08 +00:00
// F5V3 confirmed:
2020-01-11 20:13:24 +00:00
// 90 70 = HT=3, H=3, adaptive, data=no data
// 54 14 = HT=Off, H=5, adaptive, data=tube t=0,h=5
// 54 34 = HT=1, H=5, adaptive, data=tube t=1,h=5
2020-01-10 15:49:08 +00:00
// 50 70 = HT=3, H=4, adaptive, data=tube t=3,h=4
// 4C 6C = HT=3, H=3, adaptive, data=tube t=3,h=3
// 4C 4C = HT=2, H=3, adaptive, data=tube t=2,h=3
2020-01-11 20:13:24 +00:00
// 4C 2C = HT=1, H=3, adaptive, data=tube t=1,h=3
2020-01-10 15:49:08 +00:00
// 4C 0C = HT=off, H=3, adaptive, data=tube t=0,h=3
2020-01-11 20:13:24 +00:00
// 48 08 = HT=off, H=2, adaptive, data=tube t=0,h=2
// 44 04 = HT=off, H=1, adaptive, data=tube t=0,h=1
// 40 00 = HT=off,H=off, adaptive, data=tube t=0,h=0
// 34 74 = HT=3, H=5, adaptive, data=s1 (5)
// 30 70 = HT=3, H=4, adaptive, data=s1 (4)
2020-01-10 15:49:08 +00:00
// 2C 6C = HT=3, H=3, adaptive, data=s1 (3)
2020-01-11 20:13:24 +00:00
// 28 68 = HT=3, H=2, adaptive, data=s1 (2)
// 24 64 = HT=3, H=1, adaptive, data=s1 (1)
2020-01-10 15:49:08 +00:00
// F3V6 confirmed:
2020-01-11 20:13:24 +00:00
// 84 24 = HT=3, H=3, disconnect=adaptive, data=no data
2020-01-10 15:49:08 +00:00
// 50 90 = HT=4, H=4, disconnect=adaptive, data=tube t=4,h=4
2020-01-11 20:13:24 +00:00
// 44 84 = HT=4, H=1, disconnect=adaptive, data=tube t=4,h=1
// 40 80 = HT=4, H=Off,disconnect=adaptive, data=tube t=4,h=0
// 4C 6C = HT=3, H=3, disconnect=adaptive, data=tube t=3,h=3
2020-01-10 15:49:08 +00:00
// 48 68 = HT=3, H=2, disconnect=adaptive, data=tube t=3,h=2
// 44 44 = HT=2, H=1, disconnect=adaptive, data=tube t=2,h=1
// 48 28 = HT=1, H=2, disconnect=adaptive, data=tube t=1,h=2
2020-01-11 20:13:24 +00:00
// 54 14 = HT=Off,H=5, disconnect=adaptive data=tube t=0,h=5
// 34 14 = HT=3, H=5, disconnect=adaptive, data=s1 (5)
2020-01-10 15:49:08 +00:00
// 30 70 = HT=3, H=4, disconnect=adaptive, data=s1 (4)
// 2C 6C = HT=3, H=3, disconnect=adaptive, data=s1 (3)
2020-01-11 20:13:24 +00:00
// 28 08 = HT=3, H=2, disconnect=adaptive, data=s1 (2)
// 20 20 = HT=3, H=Off, disconnect=adaptive, data=s1 (0)
// 14 14 = HT=3, H=3, disconnect=fixed, data=classic (5)
2020-01-10 15:49:08 +00:00
// 10 10 = HT=3, H=4, disconnect=fixed, data=classic (4) [fixed mode appears to ignore HT bits and show HT=3 in details]
2020-01-11 20:13:24 +00:00
// 0C 0C = HT=3, H=3, disconnect=fixed, data=classic (3)
// 08 08 = HT=3, H=2, disconnect=fixed, data=classic (2)
2020-01-10 15:49:08 +00:00
// 04 64 = HT=3, H=1, disconnect=fixed, data=classic (1)
// The data is consistent among all fileVersion 3 models: F0V6, F5V3, F3V6.
//
// NOTE: F5V3 and F3V6 charts report the "Adaptive" setting as "System One" and the "Fixed"
// setting as "Classic", despite labeling the settings "Adaptive" and "Fixed" just like F0V6.
// F0V6 is consistent and labels both settings and chart as "Adaptive" and "Fixed".
//
// 400G and 502G appear to omit the humidifier settings in their details, though they
// do support humidifiers, and will show the humidification in the charts.
2019-06-14 00:31:21 +00:00
void PRS1DataChunk : : ParseHumidifierSettingV3 ( unsigned char byte1 , unsigned char byte2 , bool add_setting )
2019-06-07 20:40:26 +00:00
{
2020-01-10 15:49:08 +00:00
bool humidfixed = false ; // formerly called "Classic"
bool humidadaptive = false ; // formerly called "System One"
bool tubepresent = false ;
2019-06-07 20:40:26 +00:00
// Byte 1: 0x90 (no humidifier data), 0x50 (15ht, tube 4/5, humid 4), 0x54 (15ht, tube 5, humid 5) 0x4c (15ht, tube temp 3, humidifier 3)
2019-06-08 18:45:57 +00:00
// 0x0c (15, tube 3, humid 3, fixed)
2020-01-10 15:49:08 +00:00
// 0b1001 0000 no humidifier data
// 0b0101 0000 tube 4 and 5, humidifier 4
// 0b0101 0100 15ht, tube 5, humidifier 5
// 0b0100 1100 15ht, tube 3, humidifier 3
// 842 = humidifier status
// 1 84 = humidifier setting
// ??
2019-06-07 20:40:26 +00:00
CHECK_VALUE ( byte1 & 3 , 0 ) ;
2019-06-08 18:45:57 +00:00
int humid = byte1 > > 5 ;
switch ( humid ) {
2020-01-10 15:49:08 +00:00
case 0 : humidfixed = true ; break ; // fixed, ignores tubetemp bits and reports tubetemp=3
case 1 : humidadaptive = true ; break ; // adaptive, ignores tubetemp bits and reports tubetemp=3
case 2 : tubepresent = true ; break ; // heated tube
case 4 : break ; // no humidifier, possibly a bit flag rather than integer value, reports tubetemp=3 and humidlevel=3
2019-06-08 18:45:57 +00:00
default :
UNEXPECTED_VALUE ( humid , " known value " ) ;
break ;
}
2019-06-09 01:36:51 +00:00
bool humidifier_present = ( ( byte1 & 0x80 ) = = 0 ) ;
2019-06-07 20:40:26 +00:00
int humidlevel = ( byte1 > > 2 ) & 7 ;
// Byte 2: 0xB4 (15ht, tube 5, humid 5), 0xB0 (15ht, tube 5, humid 4), 0x90 (tube 4, humid 4), 0x6C (15ht, tube temp 3, humidifier 3)
// 0x80?
2020-01-10 15:49:08 +00:00
// 0b1011 0100 15ht, tube 5, humidifier 5
// 0b1011 0000 15ht, tube 5, humidifier 4
// 0b1001 0000 tube 4, humidifier 4
// 0b0110 1100 15ht, tube 3, humidifier 3
// 842 = tube temperature
// 1 84 = humidity level when using heated tube, thus far always identical to humidlevel
// ??
2019-06-07 20:40:26 +00:00
CHECK_VALUE ( byte2 & 3 , 0 ) ;
2020-01-10 15:49:08 +00:00
int tubehumidlevel = ( byte2 > > 2 ) & 7 ;
CHECK_VALUE ( humidlevel , tubehumidlevel ) ; // thus far always the same
int tubetemp = ( byte2 > > 5 ) & 7 ;
2019-06-09 01:36:51 +00:00
if ( humidifier_present ) {
2019-06-12 01:26:40 +00:00
if ( humidlevel > 5 | | humidlevel < 0 ) UNEXPECTED_VALUE ( humidlevel , " 0-5 " ) ; // 0=off is valid when a humidifier is attached
2019-06-09 01:36:51 +00:00
if ( humid = = 2 ) { // heated tube
2020-01-10 15:49:08 +00:00
if ( tubetemp > 5 | | tubetemp < 0 ) UNEXPECTED_VALUE ( tubetemp , " 0-5 " ) ; // TODO: maybe this is only if heated tube? 0=off is valid even in heated tube mode
2019-06-09 01:36:51 +00:00
}
}
2019-06-07 20:40:26 +00:00
2020-01-12 00:14:01 +00:00
HumidMode humidmode = HUMID_Fixed ;
if ( tubepresent ) {
humidmode = HUMID_HeatedTube ;
} else if ( humidadaptive ) {
humidmode = HUMID_Adaptive ;
}
2019-06-07 20:40:26 +00:00
if ( add_setting ) {
2020-01-12 00:14:01 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_STATUS , humidifier_present ) ) ;
if ( humidifier_present ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_MODE , humidmode ) ) ;
if ( humidmode = = HUMID_HeatedTube ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HEATED_TUBE_TEMP , tubetemp ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , tubehumidlevel ) ) ;
} else {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_HUMID_LEVEL , humidlevel ) ) ;
}
}
2019-06-07 20:40:26 +00:00
}
2020-01-10 15:49:08 +00:00
2020-01-11 20:13:24 +00:00
// Check for previously unseen data that we expect to be normal:
2020-01-10 15:49:08 +00:00
if ( family = = 0 ) {
if ( tubepresent ) {
2020-01-28 18:04:36 +00:00
// All tube temperature and humidity levels seen.
2020-01-10 15:49:08 +00:00
}
} else if ( family = = 5 ) {
if ( tubepresent ) {
2020-01-11 20:13:24 +00:00
if ( tubetemp ! = 0 & & tubetemp > 3 ) UNEXPECTED_VALUE ( tubetemp , " <= 3 " ) ;
2020-01-10 15:49:08 +00:00
}
CHECK_VALUE ( humidfixed , false ) ;
} else if ( family = = 3 ) {
if ( tubepresent ) {
2020-01-23 00:52:49 +00:00
// All tube temperature and humidity levels seen.
2020-01-10 15:49:08 +00:00
} else if ( humidadaptive ) {
2020-01-11 20:13:24 +00:00
if ( humidlevel = = 1 ) UNEXPECTED_VALUE ( humidlevel , " [0,2-5] " ) ;
2020-01-10 15:49:08 +00:00
} else if ( humidfixed ) {
2020-01-11 20:13:24 +00:00
if ( humidlevel = = 0 ) UNEXPECTED_VALUE ( humidlevel , " 1-5 " ) ;
2020-01-10 15:49:08 +00:00
}
}
2019-06-07 20:40:26 +00:00
}
2019-07-24 00:40:24 +00:00
// The below is based on a combination of the old mainblock parsing for fileVersion == 3
2019-06-07 00:19:46 +00:00
// in ParseSummary() and the switch statements of ParseSummaryF0V6.
//
// Both compliance and summary files (at least for 200X and 400X machines) seem to have
// the same length for this slice, so maybe the settings are the same? At least 0x0a
// looks like a pressure in compliance files.
bool PRS1DataChunk : : ParseSettingsF0V6 ( const unsigned char * data , int size )
{
2019-06-12 01:26:40 +00:00
static const QMap < int , int > expected_lengths = { { 0x0c , 3 } , { 0x0d , 2 } , { 0x0e , 2 } , { 0x0f , 4 } , { 0x10 , 3 } , { 0x35 , 2 } } ;
2019-06-07 00:19:46 +00:00
bool ok = true ;
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2019-09-20 19:38:14 +00:00
FlexMode flexmode = FLEX_Unknown ;
2019-06-07 00:19:46 +00:00
2019-06-12 01:26:40 +00:00
int pressure = 0 ;
2019-06-07 00:19:46 +00:00
int imin_ps = 0 ;
int imax_ps = 0 ;
int min_pressure = 0 ;
int max_pressure = 0 ;
// Parse the nested data structure which contains settings
int pos = 0 ;
do {
int code = data [ pos + + ] ;
int len = data [ pos + + ] ;
int expected_len = 1 ;
if ( expected_lengths . contains ( code ) ) {
expected_len = expected_lengths [ code ] ;
}
2019-06-09 01:36:51 +00:00
//CHECK_VALUE(len, expected_len);
2019-06-07 00:19:46 +00:00
if ( len < expected_len ) {
qWarning ( ) < < this - > sessionid < < " setting " < < code < < " too small " < < len < < " < " < < expected_len ;
ok = false ;
break ;
}
if ( pos + len > size ) {
qWarning ( ) < < this - > sessionid < < " setting " < < code < < " @ " < < pos < < " longer than remaining slice " ;
ok = false ;
break ;
}
switch ( code ) {
2019-06-11 00:30:05 +00:00
case 0 : // Device Mode
2019-06-10 18:30:50 +00:00
CHECK_VALUE ( pos , 2 ) ; // always first?
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-11 00:30:05 +00:00
switch ( data [ pos ] ) {
2019-08-29 01:30:25 +00:00
case 0 : cpapmode = PRS1_MODE_CPAP ; break ;
case 1 : cpapmode = PRS1_MODE_BILEVEL ; break ;
case 2 : cpapmode = PRS1_MODE_AUTOCPAP ; break ;
case 3 : cpapmode = PRS1_MODE_AUTOBILEVEL ; break ;
case 4 : cpapmode = PRS1_MODE_CPAPCHECK ; break ;
2019-06-11 00:30:05 +00:00
default :
UNEXPECTED_VALUE ( data [ pos ] , " known device mode " ) ;
break ;
2019-06-10 19:30:22 +00:00
}
2019-06-11 00:57:05 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
2019-06-07 00:19:46 +00:00
break ;
case 1 : // ???
2019-09-23 18:26:38 +00:00
CHECK_VALUES ( len , 1 , 2 ) ;
2019-06-13 00:37:04 +00:00
if ( data [ pos ] ! = 0 & & data [ pos ] ! = 3 ) {
CHECK_VALUES ( data [ pos ] , 1 , 2 ) ; // 1 when EZ-Start is enabled? 2 when Auto-Trial? 3 when Auto-Trial is off or Opti-Start isn't off?
2019-06-12 01:26:40 +00:00
}
2019-09-07 19:02:16 +00:00
if ( len = = 2 ) { // 400G, 500G has extra byte
2020-01-28 18:04:36 +00:00
if ( data [ pos + 1 ] ! = 0 & & data [ pos + 1 ] ! = 0x80 ) {
2019-12-07 20:10:48 +00:00
// 0x20 seen with Opti-Start enabled
// 0x30 seen with both Opti-Start and EZ-Start enabled on 500X110
2020-01-28 18:04:36 +00:00
// 0x80 seen with EZ-Start and CPAP-Check+ on 500X150
2019-12-07 20:10:48 +00:00
CHECK_VALUES ( data [ pos + 1 ] , 0x20 , 0x30 ) ;
}
2019-06-09 01:36:51 +00:00
}
2019-06-07 00:19:46 +00:00
break ;
2019-06-07 18:32:53 +00:00
case 0x0a : // CPAP pressure setting
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_CPAP ) ;
2019-06-12 01:26:40 +00:00
pressure = data [ pos ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE , pressure ) ) ;
break ;
case 0x0c : // CPAP-Check pressure setting
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 3 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_CPAPCHECK ) ;
2019-06-12 01:26:40 +00:00
min_pressure = data [ pos ] ; // Min Setting on pressure graph
max_pressure = data [ pos + 1 ] ; // Max Setting on pressure graph
pressure = data [ pos + 2 ] ; // CPAP on pressure graph and CPAP-Check Pressure on settings detail
2019-06-13 00:37:04 +00:00
// This seems to be the initial pressure. If the pressure changes mid-session, the pressure
// graph will show either the changed pressure or the majority pressure, not sure which.
// The time of change is most likely in the events file. See slice 6 for ending pressure.
//CHECK_VALUE(pressure, 0x5a);
2019-06-12 01:26:40 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE , pressure ) ) ;
2020-03-23 00:00:09 +00:00
// TODO: Once OSCAR can handle more modes, we can include these settings; right now including
// these settings makes it think this is AutoCPAP.
//this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure));
//this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure));
2019-06-07 00:19:46 +00:00
break ;
2019-06-10 18:30:50 +00:00
case 0x0d : // AutoCPAP pressure setting
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 2 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_AUTOCPAP ) ;
2019-06-10 18:30:50 +00:00
min_pressure = data [ pos ] ;
max_pressure = data [ pos + 1 ] ;
2019-06-11 00:57:05 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MIN , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MAX , max_pressure ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-10 19:30:22 +00:00
case 0x0e : // Bi-Level pressure setting
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 2 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_BILEVEL ) ;
2019-06-10 19:30:22 +00:00
min_pressure = data [ pos ] ;
max_pressure = data [ pos + 1 ] ;
2019-06-07 00:19:46 +00:00
imin_ps = max_pressure - min_pressure ;
2019-06-11 00:57:05 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP , max_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS , imin_ps ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-11 00:30:05 +00:00
case 0x0f : // Auto Bi-Level pressure setting
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 4 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_AUTOBILEVEL ) ;
2019-06-11 00:30:05 +00:00
min_pressure = data [ pos ] ;
max_pressure = data [ pos + 1 ] ;
imin_ps = data [ pos + 2 ] ;
imax_ps = data [ pos + 3 ] ;
2019-06-11 00:57:05 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MIN , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , max_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MIN , imin_ps ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MAX , imax_ps ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-12 01:26:40 +00:00
case 0x10 : // Auto-Trial mode
2019-09-26 16:37:01 +00:00
// TODO: F0V4 considers this a separate mode from CPAP or CPAPCHECK, should F0V6 as well?
// TODO: Check how auto-trial sessions are labeled in F0V6 reports.
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 3 ) ;
2019-09-19 20:43:24 +00:00
CHECK_VALUES ( cpapmode , PRS1_MODE_CPAP , PRS1_MODE_CPAPCHECK ) ; // TODO: What's the difference between auto-trial and CPAP-Check?
2019-06-13 00:37:04 +00:00
CHECK_VALUES ( data [ pos ] , 30 , 5 ) ; // Auto-Trial Duration
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_TRIAL , data [ pos ] ) ) ;
2019-06-12 01:26:40 +00:00
min_pressure = data [ pos + 1 ] ;
max_pressure = data [ pos + 2 ] ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MIN , min_pressure ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PRESSURE_MAX , max_pressure ) ) ;
break ;
case 0x2a : // EZ-Start
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-12 01:26:40 +00:00
CHECK_VALUE ( data [ pos ] , 0x80 ) ; // EZ-Start enabled
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_EZ_START , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2020-02-04 17:26:42 +00:00
case 0x42 : // EZ-Start enabled for Auto-CPAP?
2019-12-07 20:10:48 +00:00
// Seen on 500X110 before 0x2b when EZ-Start is enabled on Auto-CPAP
CHECK_VALUE ( len , 1 ) ;
2020-02-04 17:26:42 +00:00
CHECK_VALUES ( data [ pos ] , 0x00 , 0x80 ) ; // both seem to mean enabled, 0x00 appears when Opti-Start is used instead
// TODO: How to represent which one is active in practice? Should this always be "true" since
// either value means that the setting is enabled?
2019-12-07 20:10:48 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_EZ_START , data [ pos ] ! = 0 ) ) ;
break ;
2019-06-09 01:36:51 +00:00
case 0x2b : // Ramp Type
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-09 01:36:51 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ; // 0 == "Linear", 0x80 = "SmartRamp"
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TYPE , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-07 18:32:53 +00:00
case 0x2c : // Ramp Time
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-09 01:36:51 +00:00
if ( data [ pos ] ! = 0 ) { // 0 == ramp off, and ramp pressure setting doesn't appear
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , data [ pos ] ) ) ;
}
2019-06-07 00:19:46 +00:00
break ;
2019-06-07 18:32:53 +00:00
case 0x2d : // Ramp Pressure
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-08 18:21:54 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , data [ pos ] ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-09-23 16:56:06 +00:00
case 0x2e : // Flex mode
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 19:38:14 +00:00
switch ( data [ pos ] ) {
case 0 :
flexmode = FLEX_None ;
break ;
case 0x80 :
2020-01-28 17:00:22 +00:00
switch ( cpapmode ) {
case PRS1_MODE_CPAP :
2020-01-28 21:04:34 +00:00
case PRS1_MODE_CPAPCHECK :
2020-01-28 18:04:36 +00:00
case PRS1_MODE_AUTOCPAP :
2020-01-28 17:00:22 +00:00
//case PRS1_MODE_AUTOTRIAL:
flexmode = FLEX_CFlex ;
break ;
case PRS1_MODE_BILEVEL :
case PRS1_MODE_AUTOBILEVEL :
flexmode = FLEX_BiFlex ;
break ;
default :
HEX ( flexmode ) ;
UNEXPECTED_VALUE ( cpapmode , " untested mode " ) ;
break ;
}
2019-09-20 19:38:14 +00:00
break ;
case 0x90 : // C-Flex+ or A-Flex, depending on machine mode
switch ( cpapmode ) {
case PRS1_MODE_CPAP :
case PRS1_MODE_CPAPCHECK :
flexmode = FLEX_CFlexPlus ;
break ;
case PRS1_MODE_AUTOCPAP :
flexmode = FLEX_AFlex ;
break ;
default :
UNEXPECTED_VALUE ( cpapmode , " cpap or apap " ) ;
break ;
}
break ;
2019-12-04 00:04:09 +00:00
case 0xB0 : // P-Flex
2020-03-23 03:19:35 +00:00
flexmode = FLEX_PFlex ;
2020-03-23 00:00:09 +00:00
switch ( cpapmode ) {
case PRS1_MODE_AUTOCPAP :
break ;
2020-03-23 03:19:35 +00:00
default :
2020-03-23 00:00:09 +00:00
HEX ( flexmode ) ;
2020-03-23 03:19:35 +00:00
UNEXPECTED_VALUE ( cpapmode , " apap " ) ;
2020-03-23 00:00:09 +00:00
break ;
}
2019-12-04 00:04:09 +00:00
break ;
2019-09-20 19:38:14 +00:00
default :
UNEXPECTED_VALUE ( data [ pos ] , " known flex mode " ) ;
break ;
2019-06-10 18:30:50 +00:00
}
2019-09-20 19:38:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_MODE , flexmode ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-13 00:37:04 +00:00
case 0x2f : // Flex lock
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-13 00:37:04 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LOCK , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-09 01:36:51 +00:00
case 0x30 : // Flex level
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-09 01:36:51 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LEVEL , data [ pos ] ) ) ;
2020-03-23 00:00:09 +00:00
if ( flexmode = = FLEX_PFlex ) {
CHECK_VALUE ( data [ pos ] , 4 ) ; // No number appears on reports.
}
2019-06-07 00:19:46 +00:00
break ;
2019-06-07 20:40:26 +00:00
case 0x35 : // Humidifier setting
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 2 ) ;
2019-06-14 00:31:21 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos ] , data [ pos + 1 ] , true ) ;
2019-06-07 18:32:53 +00:00
break ;
2019-09-07 19:02:16 +00:00
case 0x36 : // Mask Resistance Lock
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-07 19:02:16 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
2019-09-20 04:15:40 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-08 18:21:54 +00:00
case 0x38 : // Mask Resistance
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-09 01:36:51 +00:00
if ( data [ pos ] ! = 0 ) { // 0 == mask resistance off
2020-03-23 03:19:35 +00:00
if ( data [ pos ] < 1 | | data [ pos ] > 3 ) UNEXPECTED_VALUE ( data [ pos ] , " 1-3 " ) ;
2019-09-20 04:15:40 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , data [ pos ] ) ) ;
2019-06-09 01:36:51 +00:00
}
2019-06-07 00:19:46 +00:00
break ;
2020-03-23 00:00:09 +00:00
case 0x39 : // Tubing Type Lock
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2020-03-23 00:00:09 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_TUBING_LOCK , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-09-20 01:29:33 +00:00
case 0x3b : // Tubing Type
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-10 18:30:50 +00:00
if ( data [ pos ] ! = 0 ) {
2019-09-20 16:59:14 +00:00
CHECK_VALUES ( data [ pos ] , 2 , 1 ) ; // 15HT = 2, 15 = 1, 22 = 0
2019-06-10 18:30:50 +00:00
}
2019-09-20 01:29:33 +00:00
this - > ParseTubingTypeV3 ( data [ pos ] ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-13 00:37:04 +00:00
case 0x40 : // new to 400G, also seen on 500X110, alternate tubing type? appears after 0x39 and before 0x3c
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2020-03-23 00:00:09 +00:00
if ( data [ pos ] < 0 | | data [ pos ] > 3 ) UNEXPECTED_VALUE ( data [ pos ] , " 0-3 " ) ; // 0 = 22mm, 1 = 15mm, 2 = 15HT, 3 = 12mm
2019-09-20 01:29:33 +00:00
this - > ParseTubingTypeV3 ( data [ pos ] ) ;
2019-06-09 01:36:51 +00:00
break ;
2019-09-20 01:29:33 +00:00
case 0x3c : // View Optional Screens
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 01:29:33 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_SHOW_AHI , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-09-20 01:29:33 +00:00
case 0x3e : // Auto On
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 01:29:33 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_ON , data [ pos ] ! = 0 ) ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-09-20 01:29:33 +00:00
case 0x3f : // Auto Off
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 01:29:33 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_OFF , data [ pos ] ! = 0 ) ) ;
2019-06-10 19:02:54 +00:00
break ;
case 0x43 : // new to 502G, sessions 3-8, Auto-Trial is off, Opti-Start is missing
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-10 19:02:54 +00:00
CHECK_VALUE ( data [ pos ] , 0x3C ) ;
break ;
case 0x44 : // new to 502G, sessions 3-8, Auto-Trial is off, Opti-Start is missing
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-10 19:02:54 +00:00
CHECK_VALUE ( data [ pos ] , 0xFF ) ;
2019-06-07 00:19:46 +00:00
break ;
2019-06-09 01:36:51 +00:00
case 0x45 : // new to 400G, only in last session?
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-09 01:36:51 +00:00
CHECK_VALUE ( data [ pos ] , 1 ) ;
break ;
2019-06-07 00:19:46 +00:00
default :
2020-03-23 00:00:09 +00:00
UNEXPECTED_VALUE ( code , " known setting " ) ;
2019-06-09 01:36:51 +00:00
qDebug ( ) < < " Unknown setting: " < < hex < < code < < " in " < < this - > sessionid < < " at " < < pos ;
this - > AddEvent ( new PRS1UnknownDataEvent ( QByteArray ( ( const char * ) data , size ) , pos , len ) ) ;
2019-06-07 00:19:46 +00:00
break ;
}
pos + = len ;
} while ( ok & & pos + 2 < = size ) ;
return ok ;
2019-06-06 20:08:40 +00:00
}
2019-06-08 02:05:52 +00:00
bool PRS1DataChunk : : ParseSummaryF0V6 ( void )
{
if ( this - > family ! = 0 | | this - > familyVersion ! = 6 ) {
qWarning ( ) < < " ParseSummaryF0V6 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
2019-06-10 19:02:54 +00:00
static const int minimum_sizes [ ] = { 1 , 0x2b , 9 , 4 , 2 , 4 , 1 , 4 , 0x1b , 2 , 4 , 0x0b , 1 , 2 , 6 } ;
2019-06-09 01:36:51 +00:00
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
2019-06-12 01:26:40 +00:00
// NOTE: The sizes contained in hblock can vary, even within a single machine, as can the length of hblock itself!
// TODO: hardcoding this is ugly, think of a better approach
if ( chunk_size < minimum_sizes [ 0 ] + minimum_sizes [ 1 ] + minimum_sizes [ 2 ] ) {
qWarning ( ) < < this - > sessionid < < " summary data too short: " < < chunk_size ;
return false ;
2019-06-08 02:05:52 +00:00
}
2019-06-12 01:26:40 +00:00
if ( chunk_size < 60 ) UNEXPECTED_VALUE ( chunk_size , " >= 60 " ) ;
2019-06-08 02:05:52 +00:00
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-06-08 02:05:52 +00:00
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
2019-06-09 01:36:51 +00:00
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
if ( size < minimum_sizes [ code ] ) {
2020-01-23 00:34:42 +00:00
UNEXPECTED_VALUE ( size , minimum_sizes [ code ] ) ;
2019-06-09 01:36:51 +00:00
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " too small " < < size < < " < " < < minimum_sizes [ code ] ;
2020-01-23 00:34:42 +00:00
if ( code ! = 1 ) { // Settings are variable-length, so shorter settings slices aren't fatal.
ok = false ;
break ;
}
2019-06-09 01:36:51 +00:00
}
} // else if it's past ncodes, we'll log its information below (rather than handle it)
2019-06-08 02:05:52 +00:00
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
switch ( code ) {
2019-06-09 01:36:51 +00:00
case 0 : // Equipment On
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( pos , 1 ) ; // Always first?
2019-06-09 01:36:51 +00:00
//CHECK_VALUES(data[pos], 1, 7); // or 3?
if ( size = = 4 ) { // 400G has 3 more bytes?
//CHECK_VALUE(data[pos+1], 0); // or 2, 14, 4, etc.
//CHECK_VALUES(data[pos+2], 8, 65); // or 1
//CHECK_VALUES(data[pos+3], 0, 20); // or 21, 22, etc.
}
2019-06-08 02:05:52 +00:00
break ;
case 1 : // Settings
ok = this - > ParseSettingsF0V6 ( data + pos , size ) ;
break ;
case 3 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
2019-06-14 00:31:21 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-06-08 02:05:52 +00:00
break ;
case 4 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
break ;
2019-06-10 18:30:50 +00:00
case 8 : // vs. 7 in compliance, always follows mask off (except when there's a 5, see below), also longer
// Maybe statistics of some kind, given the pressure stats that seem to appear before it on AutoCPAP machines?
2019-06-08 18:45:57 +00:00
//CHECK_VALUES(data[pos], 0x02, 0x01); // probably 16-bit value
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 1 ] , 0x00 ) ;
2019-06-10 18:30:50 +00:00
//CHECK_VALUES(data[pos+2], 0x0d, 0x0a); // probably 16-bit value, maybe OA count?
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 3 ] , 0x00 ) ;
2019-06-08 18:45:57 +00:00
//CHECK_VALUES(data[pos+4], 0x09, 0x0b); // probably 16-bit value
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 5 ] , 0x00 ) ;
2019-06-08 18:45:57 +00:00
//CHECK_VALUES(data[pos+6], 0x1e, 0x35); // probably 16-bit value
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 7 ] , 0x00 ) ;
2019-06-13 00:37:04 +00:00
//CHECK_VALUES(data[pos+8], 0x8c, 0x4c); // 16-bit value, not sure what
//CHECK_VALUE(data[pos+9], 0x00);
//CHECK_VALUES(data[pos+0xa], 0xbb, 0x00); // 16-bit minutes in large leak
//CHECK_VALUE(data[pos+0xb], 0x00);
2019-06-08 18:45:57 +00:00
//CHECK_VALUES(data[pos+0xc], 0x15, 0x02); // probably 16-bit value
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 0xd ] , 0x00 ) ;
2019-06-12 01:26:40 +00:00
//CHECK_VALUES(data[pos+0xe], 0x01, 0x00); // 16-bit VS count
//CHECK_VALUE(data[pos+0xf], 0x00);
2019-06-10 18:30:50 +00:00
//CHECK_VALUES(data[pos+0x10], 0x21, 5); // probably 16-bit value, maybe H count?
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 0x11 ] , 0x00 ) ;
2019-06-08 18:45:57 +00:00
//CHECK_VALUES(data[pos+0x12], 0x13, 0); // probably 16-bit value
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 0x13 ] , 0x00 ) ;
2019-06-10 18:30:50 +00:00
//CHECK_VALUES(data[pos+0x14], 0x05, 0); // probably 16-bit value, maybe RE count?
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 0x15 ] , 0x00 ) ;
2019-06-10 18:30:50 +00:00
//CHECK_VALUE(data[pos+0x16], 0x00, 4); // probably a 16-bit value, PB or FL count?
2019-06-08 02:05:52 +00:00
CHECK_VALUE ( data [ pos + 0x17 ] , 0x00 ) ;
2019-06-08 18:45:57 +00:00
//CHECK_VALUES(data[pos+0x18], 0x69, 0x23);
//CHECK_VALUES(data[pos+0x19], 0x44, 0x18);
//CHECK_VALUES(data[pos+0x1a], 0x80, 0x49);
2019-06-10 18:30:50 +00:00
if ( size > = 0x1f ) { // 500X is only 0x1b long!
//CHECK_VALUES(data[pos+0x1b], 0x00, 6);
CHECK_VALUE ( data [ pos + 0x1c ] , 0x00 ) ;
//CHECK_VALUES(data[pos+0x1d], 0x0c, 0x0d);
//CHECK_VALUES(data[pos+0x1e], 0x31, 0x3b);
2019-09-07 19:02:16 +00:00
// TODO: 400G and 500G has 8 more bytes?
2019-06-10 18:30:50 +00:00
// TODO: 400G sometimes has another 4 on top of that?
}
2019-06-08 02:05:52 +00:00
break ;
case 2 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
//CHECK_VALUE(data[pos+2], 0x08); // 0x01
//CHECK_VALUE(data[pos+3], 0x14); // 0x12
//CHECK_VALUE(data[pos+4], 0x01); // 0x00
//CHECK_VALUE(data[pos+5], 0x22); // 0x28
//CHECK_VALUE(data[pos+6], 0x02); // sometimes 1, 0
CHECK_VALUE ( data [ pos + 7 ] , 0x00 ) ; // 0x00
CHECK_VALUE ( data [ pos + 8 ] , 0x00 ) ; // 0x00
2019-06-09 01:36:51 +00:00
if ( size = = 0x0c ) { // 400G has 3 more bytes, seem to match Equipment On bytes
//CHECK_VALUE(data[pos+1], 0);
//CHECK_VALUES(data[pos+2], 8, 65);
//CHECK_VALUE(data[pos+3], 0);
}
2019-06-08 02:05:52 +00:00
break ;
2019-09-29 00:44:18 +00:00
case 0x0a : // Humidifier setting change
2019-06-08 02:05:52 +00:00
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
2019-06-14 00:31:21 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
2019-06-08 02:05:52 +00:00
break ;
2019-06-09 01:36:51 +00:00
case 0x0e :
2019-09-07 19:02:16 +00:00
// only seen once on 400G, many times on 500G
CHECK_VALUES ( data [ pos ] , 0 , 6 ) ;
2019-06-09 01:36:51 +00:00
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ;
2019-09-07 19:02:16 +00:00
//CHECK_VALUES(data[pos+2], 7, 9);
//CHECK_VALUES(data[pos+3], 7, 15);
//CHECK_VALUES(data[pos+4], 7, 12);
//CHECK_VALUES(data[pos+5], 0, 3);
2019-06-09 01:36:51 +00:00
break ;
2019-06-10 18:30:50 +00:00
case 0x05 :
// AutoCPAP-related? First appeared on 500X, follows 4, before 8, look like pressure values
//CHECK_VALUE(data[pos], 0x4b); // maybe min pressure? (matches ramp pressure, see ramp on pressure graph)
//CHECK_VALUE(data[pos+1], 0x5a); // maybe max pressure? (close to max on pressure graph, time at pressure graph)
//CHECK_VALUE(data[pos+2], 0x5a); // seems to match Average 90% Pressure
//CHECK_VALUE(data[pos+3], 0x58); // seems to match Average CPAP
break ;
2019-06-11 00:30:05 +00:00
case 0x07 :
// AutoBiLevel-related? First appeared on 700X, follows 4, before 8, looks like pressure values
//CHECK_VALUE(data[pos], 0x50); // maybe min IPAP or max titrated EPAP? (matches time at pressure graph, auto bi-level summary)
//CHECK_VALUE(data[pos+1], 0x64); // maybe max IPAP or max titrated IPAP? (matches time at pressure graph, auto bi-level summary)
//CHECK_VALUE(data[pos+2], 0x4b); // seems to match 90% EPAP
//CHECK_VALUE(data[pos+3], 0x64); // seems to match 90% IPAP
break ;
2019-06-12 01:26:40 +00:00
case 0x0b :
2019-09-30 14:23:28 +00:00
// CPAP-Check related, follows Mask On in CPAP-Check mode
2019-06-13 00:37:04 +00:00
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
//CHECK_VALUE(data[pos+2], 0); // probably 16-bit value
2019-06-12 01:26:40 +00:00
CHECK_VALUE ( data [ pos + 3 ] , 0 ) ;
2019-06-13 00:37:04 +00:00
//CHECK_VALUE(data[pos+4], 0); // probably 16-bit value
2019-06-12 01:26:40 +00:00
CHECK_VALUE ( data [ pos + 5 ] , 0 ) ;
2019-06-13 00:37:04 +00:00
//CHECK_VALUE(data[pos+6], 0); // probably 16-bit value
2019-06-12 01:26:40 +00:00
CHECK_VALUE ( data [ pos + 7 ] , 0 ) ;
2019-06-13 00:37:04 +00:00
//CHECK_VALUE(data[pos+8], 0); // probably 16-bit value
2019-06-12 01:26:40 +00:00
CHECK_VALUE ( data [ pos + 9 ] , 0 ) ;
2019-06-13 00:37:04 +00:00
//CHECK_VALUES(data[pos+0xa], 20, 60); // or 0? 44 when changed pressure mid-session?
2019-06-12 01:26:40 +00:00
break ;
case 0x06 :
2019-06-13 00:37:04 +00:00
// Maybe starting pressure? follows 4, before 8, looks like a pressure value, seen with CPAP-Check and EZ-Start
// Maybe ending pressure: matches ending CPAP-Check pressure if it changes mid-session.
// TODO: The daily details will show when it changed, so maybe there's an event that indicates a pressure change.
//CHECK_VALUES(data[pos], 90, 60); // maybe CPAP-Check pressure, also matches EZ-Start Pressure
2019-06-12 01:26:40 +00:00
break ;
2019-12-07 20:10:48 +00:00
case 0x0c :
// EZ-Start pressure for Auto-CPAP, seen on 500X110 following 4, before 8
// Appears to reflect the current session's EZ-Start pressure, though reported afterwards
//CHECK_VALUE(data[pos], 70, 80);
break ;
2019-06-08 02:05:52 +00:00
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
break ;
}
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-06-08 02:05:52 +00:00
this - > duration = tt ;
return ok ;
}
2019-06-14 00:31:21 +00:00
// Originally based on ParseSummaryF0V6, with changes observed in ASV sample data
// based on size, slices 0-5 look similar, and it looks like F0V6 slides 8-B are equivalent to 6-9
//
// TODO: surely there will be a way to merge these loops and abstract the machine-specific
// encodings into another function or class, but that's probably worth pursuing only after
// the details have been figured out.
bool PRS1DataChunk : : ParseSummaryF5V3 ( void )
{
if ( this - > family ! = 5 | | this - > familyVersion ! = 3 ) {
qWarning ( ) < < " ParseSummaryF5V3 called with family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
return false ;
}
const unsigned char * data = ( unsigned char * ) this - > m_data . constData ( ) ;
int chunk_size = this - > m_data . size ( ) ;
static const int minimum_sizes [ ] = { 1 , 0x35 , 9 , 4 , 2 , 4 , 0x1e , 2 , 4 , 9 } ;
static const int ncodes = sizeof ( minimum_sizes ) / sizeof ( int ) ;
// NOTE: The sizes contained in hblock can vary, even within a single machine, as can the length of hblock itself!
// TODO: hardcoding this is ugly, think of a better approach
if ( chunk_size < minimum_sizes [ 0 ] + minimum_sizes [ 1 ] + minimum_sizes [ 2 ] ) {
qWarning ( ) < < this - > sessionid < < " summary data too short: " < < chunk_size ;
return false ;
}
2019-06-14 02:18:49 +00:00
// We've once seen a short summary with no mask-on/off: just equipment-on, settings, 9, equipment-off
2020-03-09 17:57:43 +00:00
// (And we've seen something similar in F3V6.)
2019-06-14 02:18:49 +00:00
if ( chunk_size < 75 ) UNEXPECTED_VALUE ( chunk_size , " >= 75 " ) ;
2019-06-14 00:31:21 +00:00
bool ok = true ;
int pos = 0 ;
int code , size ;
int tt = 0 ;
2019-12-01 22:30:39 +00:00
while ( ok & & pos < chunk_size ) {
2019-06-14 00:31:21 +00:00
code = data [ pos + + ] ;
if ( ! this - > hblock . contains ( code ) ) {
qWarning ( ) < < this - > sessionid < < " missing hblock entry for " < < code ;
ok = false ;
break ;
}
size = this - > hblock [ code ] ;
if ( code < ncodes ) {
// make sure the handlers below don't go past the end of the buffer
if ( size < minimum_sizes [ code ] ) {
2020-01-23 00:34:42 +00:00
UNEXPECTED_VALUE ( size , minimum_sizes [ code ] ) ;
2019-06-14 00:31:21 +00:00
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " too small " < < size < < " < " < < minimum_sizes [ code ] ;
2020-01-23 00:34:42 +00:00
if ( code ! = 1 ) { // Settings are variable-length, so shorter settings slices aren't fatal.
ok = false ;
break ;
}
2019-06-14 00:31:21 +00:00
}
} // else if it's past ncodes, we'll log its information below (rather than handle it)
if ( pos + size > chunk_size ) {
qWarning ( ) < < this - > sessionid < < " slice " < < code < < " @ " < < pos < < " longer than remaining chunk " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 : // Equipment On
CHECK_VALUE ( pos , 1 ) ; // Always first?
2019-09-19 18:21:12 +00:00
//CHECK_VALUES(data[pos], 1, 7); // or 3, or 0? 3 when machine turned on via auto-on, 1 when turned on via button
2019-06-14 00:31:21 +00:00
CHECK_VALUE ( size , 1 ) ;
break ;
case 1 : // Settings
ok = this - > ParseSettingsF5V3 ( data + pos , size ) ;
break ;
case 9 : // new to F5V3 vs. F0V6, comes right after settings, before mask on?
CHECK_VALUE ( data [ pos ] , 0 ) ;
CHECK_VALUE ( data [ pos + 1 ] , 1 ) ;
CHECK_VALUE ( data [ pos + 2 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 3 ] , 1 ) ;
CHECK_VALUE ( data [ pos + 4 ] , 1 ) ;
CHECK_VALUE ( data [ pos + 5 ] , 0 ) ;
CHECK_VALUE ( data [ pos + 6 ] , 2 ) ;
CHECK_VALUE ( data [ pos + 7 ] , 1 ) ;
2019-07-23 16:52:41 +00:00
CHECK_VALUES ( data [ pos + 8 ] , 0 , 1 ) ; // 1 = patient disconnect alarm of 15 sec, not sure where time is encoded
2019-09-20 16:59:14 +00:00
if ( data [ pos + 8 ] ! = 0 ) {
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_DISCONNECT_ALARM , data [ pos + 8 ] * 15 ) ) ;
}
CHECK_VALUE ( size , 9 ) ;
2019-06-14 00:31:21 +00:00
break ;
case 3 : // Mask On
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOn ) ) ;
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
break ;
case 4 : // Mask Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , MaskOff ) ) ;
break ;
2019-06-14 01:29:43 +00:00
case 5 : // ASV pressure stats per mask-on slice
//CHECK_VALUE(data[pos], 0x28); // 90% EPAP
//CHECK_VALUE(data[pos+1], 0x23); // average EPAP
//CHECK_VALUE(data[pos+2], 0x24); // 90% PS
//CHECK_VALUE(data[pos+3], 0x17); // average PS
break ;
case 6 : // Patient statistics per mask-on slice
// These get averaged on a time-weighted basis in the final report.
// Where is H count?
2019-06-14 02:18:49 +00:00
//CHECK_VALUE(data[pos], 0x00); // probably 16-bit value
2019-06-14 00:31:21 +00:00
CHECK_VALUE ( data [ pos + 1 ] , 0x00 ) ;
2019-06-14 01:29:43 +00:00
//CHECK_VALUE(data[pos+2], 0x00); // 16-bit OA count
//CHECK_VALUE(data[pos+3], 0x00);
2019-06-14 02:18:49 +00:00
//CHECK_VALUE(data[pos+4], 0x00); // probably 16-bit value
2019-06-14 00:31:21 +00:00
CHECK_VALUE ( data [ pos + 5 ] , 0x00 ) ;
2019-06-14 01:29:43 +00:00
//CHECK_VALUE(data[pos+6], 0x00); // 16-bit CA count
//CHECK_VALUE(data[pos+7], 0x00);
//CHECK_VALUE(data[pos+8], 0x00); // 16-bit minutes in LL
//CHECK_VALUE(data[pos+9], 0x00);
//CHECK_VALUE(data[pos+0xa], 0x0f); // 16-bit minutes in PB
//CHECK_VALUE(data[pos+0xb], 0x00);
//CHECK_VALUE(data[pos+0xc], 0x14); // 16-bit VS count
//CHECK_VALUE(data[pos+0xd], 0x00);
2019-06-19 21:28:42 +00:00
//CHECK_VALUE(data[pos+0xe], 0x05); // 16-bit H count for type 0xd
//CHECK_VALUE(data[pos+0xf], 0x00);
//CHECK_VALUE(data[pos+0x10], 0x00); // 16-bit H count for type 7
//CHECK_VALUE(data[pos+0x11], 0x00);
2019-06-14 01:29:43 +00:00
//CHECK_VALUE(data[pos+0x12], 0x02); // 16-bit FL count
//CHECK_VALUE(data[pos+0x13], 0x00);
//CHECK_VALUE(data[pos+0x14], 0x28); // 0x69 (105)
//CHECK_VALUE(data[pos+0x15], 0x17); // average total leak
//CHECK_VALUE(data[pos+0x16], 0x5b); // 0x7d (125)
2019-06-19 21:28:42 +00:00
//CHECK_VALUE(data[pos+0x17], 0x09); // 16-bit H count for type 0xe
//CHECK_VALUE(data[pos+0x18], 0x00);
2019-06-14 01:29:43 +00:00
//CHECK_VALUE(data[pos+0x19], 0x10); // average breath rate
//CHECK_VALUE(data[pos+0x1a], 0x2d); // average TV / 10
//CHECK_VALUE(data[pos+0x1b], 0x63); // average % PTB
//CHECK_VALUE(data[pos+0x1c], 0x07); // average minute vent
2019-06-19 21:28:42 +00:00
//CHECK_VALUE(data[pos+0x1d], 0x06); // average leak
2019-06-14 00:31:21 +00:00
break ;
case 2 : // Equipment Off
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ;
this - > AddEvent ( new PRS1ParsedSliceEvent ( tt , EquipmentOff ) ) ;
2019-06-14 01:29:43 +00:00
//CHECK_VALUE(data[pos+2], 0x01); // 0x08
//CHECK_VALUE(data[pos+3], 0x17); // 0x16, 0x18
2019-06-14 02:18:49 +00:00
//CHECK_VALUE(data[pos+4], 0x00);
2019-06-14 01:29:43 +00:00
//CHECK_VALUE(data[pos+5], 0x29); // 0x2a, 0x28, 0x26, 0x36
//CHECK_VALUE(data[pos+6], 0x01); // 0x00
2019-06-14 00:31:21 +00:00
CHECK_VALUE ( data [ pos + 7 ] , 0x00 ) ;
CHECK_VALUE ( data [ pos + 8 ] , 0x00 ) ;
break ;
case 8 : // Humidier setting change
tt + = data [ pos ] | ( data [ pos + 1 ] < < 8 ) ; // This adds to the total duration (otherwise it won't match report)
this - > ParseHumidifierSettingV3 ( data [ pos + 2 ] , data [ pos + 3 ] ) ;
break ;
default :
UNEXPECTED_VALUE ( code , " known slice code " ) ;
break ;
}
pos + = size ;
2019-12-01 22:30:39 +00:00
}
2019-06-14 00:31:21 +00:00
this - > duration = tt ;
return ok ;
}
// Based initially on ParseSettingsF0V6. Many of the codes look the same, like always starting with 0, 0x35 looking like
// a humidifier setting, etc., but the contents are sometimes a bit different, such as mode values and pressure settings.
//
// new settings to find: breath rate, tubing lock, alarms,
bool PRS1DataChunk : : ParseSettingsF5V3 ( const unsigned char * data , int size )
{
static const QMap < int , int > expected_lengths = { { 0x0a , 5 } , /*{0x0c,3}, {0x0d,2}, {0x0e,2}, {0x0f,4}, {0x10,3},*/ { 0x14 , 3 } , { 0x2e , 2 } , { 0x35 , 2 } } ;
bool ok = true ;
2019-08-29 01:30:25 +00:00
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN ;
2019-06-14 00:31:21 +00:00
2019-06-16 00:56:55 +00:00
// F5V3 uses a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
static const float GAIN = 0.125 ; // TODO: parameterize this somewhere better
2019-06-14 00:31:21 +00:00
int max_pressure = 0 ;
int min_ps = 0 ;
int max_ps = 0 ;
int min_epap = 0 ;
int max_epap = 0 ;
2020-03-23 00:00:09 +00:00
int breath_rate ;
int timed_inspiration ;
2019-06-14 00:31:21 +00:00
// Parse the nested data structure which contains settings
int pos = 0 ;
do {
int code = data [ pos + + ] ;
int len = data [ pos + + ] ;
int expected_len = 1 ;
if ( expected_lengths . contains ( code ) ) {
expected_len = expected_lengths [ code ] ;
}
//CHECK_VALUE(len, expected_len);
if ( len < expected_len ) {
qWarning ( ) < < this - > sessionid < < " setting " < < code < < " too small " < < len < < " < " < < expected_len ;
ok = false ;
break ;
}
if ( pos + len > size ) {
qWarning ( ) < < this - > sessionid < < " setting " < < code < < " @ " < < pos < < " longer than remaining slice " ;
ok = false ;
break ;
}
switch ( code ) {
case 0 : // Device Mode
CHECK_VALUE ( pos , 2 ) ; // always first?
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 00:31:21 +00:00
switch ( data [ pos ] ) {
2019-08-29 01:30:25 +00:00
case 0 : cpapmode = PRS1_MODE_ASV ; break ;
2019-06-14 00:31:21 +00:00
default :
UNEXPECTED_VALUE ( data [ pos ] , " known device mode " ) ;
break ;
}
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_CPAP_MODE , ( int ) cpapmode ) ) ;
break ;
case 1 : // ???
2019-09-23 18:26:38 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 00:31:21 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 1 ) ; // 1 when when Opti-Start is on? 0 when off?
/*
if ( data [ pos ] ! = 0 & & data [ pos ] ! = 3 ) {
CHECK_VALUES ( data [ pos ] , 1 , 2 ) ; // 1 when EZ-Start is enabled? 2 when Auto-Trial? 3 when Auto-Trial is off or Opti-Start isn't off?
}
*/
break ;
case 0x0a : // ASV with variable EPAP pressure setting
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 5 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_ASV ) ;
2019-06-14 00:31:21 +00:00
max_pressure = data [ pos ] ;
min_epap = data [ pos + 1 ] ;
max_epap = data [ pos + 2 ] ;
min_ps = data [ pos + 3 ] ;
max_ps = data [ pos + 4 ] ;
2019-06-16 00:56:55 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MIN , min_epap , GAIN ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_EPAP_MAX , max_epap , GAIN ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MIN , min_epap + min_ps , GAIN ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_IPAP_MAX , qMin ( max_pressure , max_epap + max_ps ) , GAIN ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MIN , min_ps , GAIN ) ) ;
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_PS_MAX , max_ps , GAIN ) ) ;
2019-06-14 00:31:21 +00:00
break ;
2019-07-23 16:52:41 +00:00
case 0x14 : // ASV backup rate
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 3 ) ;
2019-08-29 01:30:25 +00:00
CHECK_VALUE ( cpapmode , PRS1_MODE_ASV ) ;
2019-09-23 18:10:23 +00:00
switch ( data [ pos ] ) {
//case 0: // Breath Rate Off in F3V6 setting 0x1e
case 1 : // Breath Rate Auto
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Auto ) ) ;
CHECK_VALUE ( data [ pos + 1 ] , 0 ) ; // 0 for auto
CHECK_VALUE ( data [ pos + 2 ] , 0 ) ; // 0 for auto
break ;
case 2 : // Breath Rate (fixed BPM)
2020-03-23 00:00:09 +00:00
breath_rate = data [ pos + 1 ] ;
timed_inspiration = data [ pos + 2 ] ;
2020-03-23 03:19:35 +00:00
CHECK_VALUE ( breath_rate , 10 ) ;
CHECK_VALUE ( timed_inspiration , 24 ) ;
2019-09-23 18:10:23 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_MODE , PRS1Backup_Fixed ) ) ;
2020-03-23 00:00:09 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_BACKUP_BREATH_RATE , breath_rate ) ) ; // BPM
this - > AddEvent ( new PRS1ScaledSettingEvent ( PRS1_SETTING_BACKUP_TIMED_INSPIRATION , timed_inspiration , 0.1 ) ) ;
2019-09-23 18:10:23 +00:00
break ;
default :
CHECK_VALUES ( data [ pos ] , 1 , 2 ) ; // 1 = auto, 2 = fixed BPM (0 = off in F3V6 setting 0x1e)
break ;
}
2019-06-14 00:31:21 +00:00
break ;
/*
case 0x2a : // EZ-Start
CHECK_VALUE ( data [ pos ] , 0x80 ) ; // EZ-Start enabled
break ;
*/
case 0x2b : // Ramp Type
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 02:18:49 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ; // 0 == "Linear", 0x80 = "SmartRamp"
2019-09-20 16:59:14 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TYPE , data [ pos ] ! = 0 ) ) ;
2019-06-14 00:31:21 +00:00
break ;
case 0x2c : // Ramp Time
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 00:31:21 +00:00
if ( data [ pos ] ! = 0 ) { // 0 == ramp off, and ramp pressure setting doesn't appear
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RAMP_TIME , data [ pos ] ) ) ;
}
break ;
case 0x2d : // Ramp Pressure (with ASV pressure encoding)
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-16 00:56:55 +00:00
this - > AddEvent ( new PRS1PressureSettingEvent ( PRS1_SETTING_RAMP_PRESSURE , data [ pos ] , GAIN ) ) ;
2019-06-14 00:31:21 +00:00
break ;
2019-09-23 16:56:06 +00:00
case 0x2e : // Flex mode and level (ASV variant)
CHECK_VALUE ( len , 2 ) ;
switch ( data [ pos ] ) {
case 0 : // Bi-Flex
// [0x00, N] for Bi-Flex level N
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_MODE , FLEX_BiFlex ) ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_FLEX_LEVEL , data [ pos + 1 ] ) ) ;
break ;
case 0x20 : // Rise Time
// [0x20, 0x03] for no flex, rise time setting = 3, no rise lock
2020-01-12 21:53:43 +00:00
CHECK_VALUE ( data [ pos + 1 ] , 3 ) ;
2019-09-23 16:56:06 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_RISE_TIME , data [ pos + 1 ] ) ) ;
break ;
default :
CHECK_VALUES ( data [ pos ] , 0 , 0x20 ) ;
break ;
2019-06-14 00:31:21 +00:00
}
break ;
case 0x2f : // Flex lock? (was on F0V6, 0x80 for locked)
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 00:31:21 +00:00
CHECK_VALUE ( data [ pos ] , 0 ) ;
2019-09-20 16:59:14 +00:00
//this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_LOCK, data[pos] != 0));
2019-06-14 00:31:21 +00:00
break ;
2019-09-23 16:56:06 +00:00
//case 0x30: ASV puts the flex level in the 0x2e setting for some reason
2019-06-14 00:31:21 +00:00
case 0x35 : // Humidifier setting
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 2 ) ;
2019-06-14 00:31:21 +00:00
this - > ParseHumidifierSettingV3 ( data [ pos ] , data [ pos + 1 ] , true ) ;
break ;
2019-06-14 02:18:49 +00:00
case 0x36 : // Mask Resistance Lock
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 02:18:49 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ; // 0x80 = locked
2019-09-20 04:15:40 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_LOCK , data [ pos ] ! = 0 ) ) ;
2019-06-14 00:31:21 +00:00
break ;
2019-06-14 02:18:49 +00:00
case 0x38 : // Mask Resistance
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 00:31:21 +00:00
if ( data [ pos ] ! = 0 ) { // 0 == mask resistance off
2019-09-20 04:15:40 +00:00
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_MASK_RESIST_SETTING , data [ pos ] ) ) ;
2019-06-14 00:31:21 +00:00
}
break ;
case 0x39 :
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-06-14 00:31:21 +00:00
CHECK_VALUE ( data [ pos ] , 0 ) ; // 0x80 maybe auto-trial in F0V6?
break ;
2019-06-14 02:18:49 +00:00
case 0x3b : // Tubing Type
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2020-03-23 00:00:09 +00:00
if ( data [ pos ] < 0 | | data [ pos ] > 2 ) UNEXPECTED_VALUE ( data [ pos ] , " 0-2 " ) ; // 15HT = 2, 15 = 1, 22 = 0, though report only says "15" for 15HT
2019-09-20 01:29:33 +00:00
this - > ParseTubingTypeV3 ( data [ pos ] ) ;
2019-06-14 00:31:21 +00:00
break ;
2019-09-20 01:29:33 +00:00
case 0x3c : // View Optional Screens
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 01:29:33 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_SHOW_AHI , data [ pos ] ! = 0 ) ) ;
2019-06-14 00:31:21 +00:00
break ;
2019-09-20 01:29:33 +00:00
case 0x3d : // Auto On (ASV variant)
2019-09-23 16:56:06 +00:00
CHECK_VALUE ( len , 1 ) ;
2019-09-20 01:29:33 +00:00
CHECK_VALUES ( data [ pos ] , 0 , 0x80 ) ;
this - > AddEvent ( new PRS1ParsedSettingEvent ( PRS1_SETTING_AUTO_ON , data [ pos ] ! = 0 ) ) ;
2019-06-14 00:31:21 +00:00
break ;
default :
2020-03-23 00:00:09 +00:00
UNEXPECTED_VALUE ( code , " known setting " ) ;
2019-06-14 00:31:21 +00:00
qDebug ( ) < < " Unknown setting: " < < hex < < code < < " in " < < this - > sessionid < < " at " < < pos ;
this - > AddEvent ( new PRS1UnknownDataEvent ( QByteArray ( ( const char * ) data , size ) , pos , len ) ) ;
break ;
}
pos + = len ;
} while ( ok & & pos + 2 < = size ) ;
return ok ;
}
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
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
2019-08-29 01:30:25 +00:00
CPAPMode cpapmode = MODE_UNKNOWN ;
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 :
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 ( ) ;
2019-08-29 01:30:25 +00:00
if ( cpapmode = = MODE_CPAP ) { // Auto-Trial is reported as CPAP but with a minimum and maximum pressure,
cpapmode = MODE_APAP ; // so import it as APAP, since that's what it's really doing.
}
2019-09-26 16:37:01 +00:00
// TODO: what about CPAPCHECK?
2019-05-27 14:28:14 +00:00
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 ( ) ;
2019-08-29 01:30:25 +00:00
if ( cpapmode = = MODE_BILEVEL_FIXED ) {
cpapmode = MODE_BILEVEL_AUTO_VARIABLE_PS ; // TODO: this isn't quite right, on ventilators it's actually fixed EPAP with variable PS
}
2019-05-27 14:28:14 +00:00
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 :
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 ;
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 ;
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 :
case PRS1_SETTING_BACKUP_BREATH_RATE :
case PRS1_SETTING_BACKUP_TIMED_INSPIRATION :
2019-09-20 16:59:14 +00:00
case PRS1_SETTING_TIDAL_VOLUME :
case PRS1_SETTING_AUTO_TRIAL :
case PRS1_SETTING_EZ_START :
2019-09-23 16:39:20 +00:00
case PRS1_SETTING_RISE_TIME :
2020-03-23 00:00:09 +00:00
case PRS1_SETTING_RISE_TIME_LOCK :
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 :
//TODO: define and add new channels for any of these that we want to import
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-08-29 01:30:25 +00:00
session - > settings [ CPAP_Mode ] = cpapmode ;
2019-06-05 21:08:45 +00:00
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-05-27 15:14:55 +00:00
bool PRS1DataChunk : : ParseSummary ( )
2019-05-27 14:25:57 +00:00
{
2019-05-27 15:14:55 +00:00
switch ( this - > family ) {
2014-08-03 13:00:13 +00:00
case 0 :
2019-05-27 15:14:55 +00:00
if ( this - > familyVersion = = 6 ) {
return this - > ParseSummaryF0V6 ( ) ;
} else if ( this - > familyVersion = = 4 ) {
return this - > ParseSummaryF0V4 ( ) ;
2014-08-04 15:50:19 +00:00
} else {
2019-05-27 15:14:55 +00:00
return this - > ParseSummaryF0V23 ( ) ;
2014-08-04 15:50:19 +00:00
}
2014-08-03 13:00:13 +00:00
case 3 :
2019-07-24 00:40:24 +00:00
if ( this - > familyVersion = = 6 ) {
return this - > ParseSummaryF3V6 ( ) ;
2019-07-24 00:54:39 +00:00
} else if ( this - > familyVersion = = 3 ) {
return this - > ParseSummaryF3V3 ( ) ;
2019-07-24 00:40:24 +00:00
}
2014-09-29 14:41:31 +00:00
break ;
2014-08-03 13:00:13 +00:00
case 5 :
2019-05-27 15:14:55 +00:00
if ( this - > familyVersion = = 1 ) {
return this - > ParseSummaryF5V012 ( ) ;
} else if ( this - > familyVersion = = 0 ) {
return this - > ParseSummaryF5V012 ( ) ;
} else if ( this - > familyVersion = = 2 ) {
return this - > ParseSummaryF5V012 ( ) ;
} else if ( this - > familyVersion = = 3 ) {
return this - > ParseSummaryF5V3 ( ) ;
2014-08-17 12:56:05 +00:00
}
2016-01-19 04:26:28 +00:00
default :
;
2014-08-03 13:00:13 +00:00
}
2019-05-27 15:14:55 +00:00
qWarning ( ) < < " unexpected family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
2014-09-29 14:41:31 +00:00
return false ;
2014-05-31 21:25:07 +00:00
}
2014-04-17 05:58:57 +00:00
2019-05-27 14:25:57 +00:00
2019-10-09 17:35:02 +00:00
// TODO: The nested switch statement below just begs for per-version subclasses.
bool PRS1DataChunk : : ParseEvents ( )
2019-05-29 15:20:20 +00:00
{
bool ok = false ;
switch ( this - > family ) {
case 0 :
2019-10-09 17:35:02 +00:00
switch ( this - > familyVersion ) {
case 2 : ok = this - > ParseEventsF0V23 ( ) ; break ;
case 3 : ok = this - > ParseEventsF0V23 ( ) ; break ;
case 4 : ok = this - > ParseEventsF0V4 ( ) ; break ;
case 6 : ok = this - > ParseEventsF0V6 ( ) ; break ;
2019-08-05 00:36:40 +00:00
}
2019-05-29 15:20:20 +00:00
break ;
case 3 :
2019-10-09 17:35:02 +00:00
switch ( this - > familyVersion ) {
case 3 : ok = this - > ParseEventsF3V3 ( ) ; break ;
case 6 : ok = this - > ParseEventsF3V6 ( ) ; break ;
2019-05-29 15:20:20 +00:00
}
break ;
case 5 :
2019-10-05 00:10:35 +00:00
switch ( this - > familyVersion ) {
case 0 : ok = this - > ParseEventsF5V0 ( ) ; break ;
case 1 : ok = this - > ParseEventsF5V1 ( ) ; break ;
case 2 : ok = this - > ParseEventsF5V2 ( ) ; break ;
case 3 : ok = this - > ParseEventsF5V3 ( ) ; break ;
2019-05-29 15:20:20 +00:00
}
break ;
default :
qDebug ( ) < < " Unknown PRS1 family " < < this - > family < < " familyVersion " < < this - > familyVersion ;
}
return ok ;
}
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 ) {
if ( slice . status = = MaskOn & & slice . end > slice . start ) {
maskOn . append ( slice ) ;
}
}
// 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 ) {
2019-11-19 19:18:13 +00:00
int offset = 0 ;
// F3V3 sometimes omits the (empty) first event chunk if the first slice is
// shorter than 2 minutes.
if ( m_event_chunks . first ( ) - > family = = 3 & & m_event_chunks . first ( ) - > familyVersion = = 3 ) {
offset = maskOn . count ( ) - m_event_chunks . count ( ) ;
if ( offset < 0 ) {
qCritical ( ) < < sessionid < < " has more event chunks than mask-on slices! " ;
offset = 0 ; // avoid out-of-bounds references below
}
}
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 ] ;
2019-11-19 19:18:13 +00:00
if ( eventlists . count ( ) + offset ! = maskOn . count ( ) ) {
qWarning ( ) < < sessionid < < " has " < < maskOn . 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
2019-11-19 19:18:13 +00:00
int j = i + offset ;
if ( eventlists [ i ] - > first ( ) < maskOn [ j ] . start | | eventlists [ i ] - > first ( ) > maskOn [ j ] . end | |
eventlists [ i ] - > last ( ) < maskOn [ j ] . start | | eventlists [ i ] - > last ( ) > maskOn [ j ] . end ) {
qWarning ( ) < < sessionid < < " channel " < < * pChannelID < < " has events outside of mask-on slice " < < i ;
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 ) ;
qint32 sid = session_s . toInt ( & numeric , m_sessionid_base ) ;
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 ;
2020-01-21 18:49:02 +00:00
qint64 ignoreBefore = p_profile - > session - > ignoreOlderSessionsDate ( ) . toMSecsSinceEpoch ( ) / 1000 ;
2020-01-19 01:16:31 +00:00
bool ignoreOldSessions = p_profile - > session - > ignoreOlderSessions ( ) ;
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-01-28 21:04:34 +00:00
void PRS1Import : : ParseOximetry ( )
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 ;
// Split eventlist on invalid values (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 ] ;
bool valid = ( value ! = 255 ) ;
if ( valid ) {
if ( pending_samples = = false ) {
pending_samples = true ;
start_i = i ;
start_ti = ti ;
}
if ( channel = = OXI_Pulse ) {
if ( value > 200 ) UNEXPECTED_VALUE ( value , " <= 200 bpm " ) ;
} 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-01-28 21:04:34 +00:00
void PRS1Import : : ParseWaveforms ( )
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.
//
// Evidently the machines' 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
// their notion of the correct time, since the machines only record time at 1-second
// 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
2019-07-25 02:42:00 +00:00
if ( ( waveform - > family = = 5 & & waveform - > familyVersion = = 3 ) | |
( waveform - > family = = 3 & & waveform - > familyVersion = = 6 ) ) {
// F5V3 and F3V6 use a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
2019-06-16 00:56:55 +00:00
pressure_gain = 0.125F ; // TODO: this should be parameterized somewhere better, once we have a clear idea of which machines use this
}
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 ) {
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
{
2014-09-29 14:41:31 +00:00
if ( mach - > unsupported ( ) )
return ;
2019-05-03 18:45:21 +00:00
if ( ParseSession ( ) ) {
SaveSessionToDatabase ( ) ;
}
}
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 ;
2014-08-04 15:40:56 +00:00
session = new Session ( mach , 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.
if ( session - > m_slices . count ( ) = = 0 | | ( m_event_chunks . count ( ) = = 0 & & m_wavefiles . isEmpty ( ) & & oxifile . isEmpty ( ) ) ) {
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-01-28 21:04:34 +00:00
ImportWaveforms ( ) ;
2018-05-05 07:14:44 +00:00
}
2014-08-04 15:40:56 +00:00
2019-06-05 14:24:32 +00:00
if ( ! oxifile . isEmpty ( ) ) {
// Parse .006 Waveform file
oximetry = loader - > ParseFile ( oxifile ) ;
oximetry = CoalesceWaveformChunks ( oximetry ) ;
2020-01-28 21:04:34 +00:00
ParseOximetry ( ) ;
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-01-28 21:04:34 +00:00
void PRS1Import : : ImportWaveforms ( )
2019-10-26 01:17:41 +00:00
{
QMap < qint64 , PRS1DataChunk * > waveform_chunks ;
if ( m_wavefiles . count ( ) > 1 ) {
qDebug ( ) < < session - > session ( ) < < " Waveform data split across multiple files " ;
}
for ( auto & f : m_wavefiles ) {
// Parse a single .005 Waveform file
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.
waveforms = waveform_chunks . values ( ) ;
// Coalesce contiguous waveform chunks into larger chunks.
waveforms = CoalesceWaveformChunks ( waveforms ) ;
if ( session - > eventlist . contains ( CPAP_FlowRate ) ) {
if ( waveforms . size ( ) > 0 ) {
// Delete anything called "Flow rate" picked up in the events file if real data is present
qWarning ( ) < < session - > session ( ) < < " Deleting flow rate events due to flow rate waveform data " ;
session - > destroyEvent ( CPAP_FlowRate ) ;
}
}
// Extract raw data into channels.
2020-01-28 21:04:34 +00:00
ParseWaveforms ( ) ;
2019-10-26 01:17:41 +00:00
}
2019-05-03 18:45:21 +00:00
void PRS1Import : : SaveSessionToDatabase ( void )
{
// Make sure it's saved
session - > SetChanged ( true ) ;
2011-07-21 11:02:23 +00:00
2019-05-03 18:45:21 +00:00
// Add the session to the database
loader - > addSession ( session ) ;
2014-04-17 05:58:57 +00:00
2019-05-03 18:45:21 +00:00
// Update indexes, process waveform and perform flagging
session - > UpdateSummaries ( ) ;
2014-04-17 05:58:57 +00:00
2019-05-03 18:45:21 +00:00
// Save is not threadsafe
loader - > saveMutex . lock ( ) ;
session - > Store ( mach - > getDataPath ( ) ) ;
loader - > saveMutex . unlock ( ) ;
2011-07-27 09:21:53 +00:00
2019-05-03 18:45:21 +00:00
// Unload them from memory
session - > TrashEvents ( ) ;
2014-05-31 21:25:07 +00:00
}
2011-07-27 09:21:53 +00:00
2018-03-23 19:24:29 +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 ;
}
PRS1DataChunk * chunk = nullptr , * lastchunk = nullptr ;
2019-05-14 22:57:04 +00:00
int cnt = 0 ;
do {
2020-01-05 01:47:28 +00:00
chunk = PRS1DataChunk : : ParseNext ( f , 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 ;
LogUnexpectedMessage ( message ) ;
// 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 + + ;
} while ( ! f . atEnd ( ) ) ;
return CHUNKS ;
}
2020-01-05 01:47:28 +00:00
PRS1DataChunk : : PRS1DataChunk ( QFile & f , PRS1Loader * in_loader ) : loader ( in_loader )
2019-05-15 02:49:41 +00:00
{
m_path = QFileInfo ( f ) . canonicalFilePath ( ) ;
}
2019-05-22 15:00:45 +00:00
PRS1DataChunk : : ~ PRS1DataChunk ( )
{
for ( int i = 0 ; i < m_parsedData . count ( ) ; i + + ) {
PRS1ParsedEvent * e = m_parsedData . at ( i ) ;
delete e ;
}
}
2019-05-15 02:49:41 +00:00
2020-01-05 01:47:28 +00:00
PRS1DataChunk * PRS1DataChunk : : ParseNext ( QFile & f , PRS1Loader * loader )
2019-05-14 22:57:04 +00:00
{
PRS1DataChunk * out_chunk = nullptr ;
2020-01-05 01:47:28 +00:00
PRS1DataChunk * chunk = new PRS1DataChunk ( f , loader ) ;
2019-05-14 23:57:01 +00:00
2018-03-23 19:24:29 +00:00
do {
2019-05-15 12:36:31 +00:00
// Parse the header and calculate its checksum.
2019-05-15 02:49:41 +00:00
bool ok = chunk - > ReadHeader ( f ) ;
2019-05-15 12:36:31 +00:00
if ( ! ok ) {
break ;
}
// Make sure the calculated checksum matches the stored checksum.
if ( chunk - > calcChecksum ! = chunk - > storedChecksum ) {
qWarning ( ) < < chunk - > m_path < < " header checksum calc " < < chunk - > calcChecksum < < " != stored " < < chunk - > storedChecksum ;
break ;
}
// Read the block's data and calculate the block CRC.
ok = chunk - > ReadData ( f ) ;
if ( ! ok ) {
break ;
}
2019-05-15 19:16:14 +00:00
// Make sure the calculated CRC over the entire chunk (header and data) matches the stored CRC.
if ( chunk - > calcCrc ! = chunk - > storedCrc ) {
2020-01-28 14:34:02 +00:00
// Corrupt data block, warn about it.
2019-06-04 02:01:02 +00:00
qWarning ( ) < < chunk - > m_path < < " @ " < < chunk - > m_filepos < < " block CRC calc " < < hex < < chunk - > calcCrc < < " != stored " < < hex < < chunk - > storedCrc ;
2019-12-02 22:30:28 +00:00
// TODO: When this happens, it's usually because the chunk was truncated and another chunk header
// exists within the blockSize bytes. In theory it should be possible to rewing and resync by
// looking for another chunk header with the same fileVersion, htype, family, familyVersion, and
// ext (blockSize and other fields could vary).
//
// But this is quite rare, so for now we bail on the rest of the file.
2019-06-04 02:01:02 +00:00
break ;
2019-05-15 19:16:14 +00:00
}
2019-05-15 02:49:41 +00:00
// Only return the chunk if it has passed all tests above.
out_chunk = chunk ;
} while ( false ) ;
if ( out_chunk = = nullptr ) delete chunk ;
return out_chunk ;
}
bool PRS1DataChunk : : ReadHeader ( QFile & f )
{
bool ok = false ;
do {
2019-05-15 19:16:14 +00:00
// Read common header fields.
2019-05-15 02:49:41 +00:00
this - > m_filepos = f . pos ( ) ;
this - > m_header = f . read ( 15 ) ;
if ( this - > m_header . size ( ) ! = 15 ) {
2019-09-19 14:33:23 +00:00
if ( this - > m_header . size ( ) = = 0 ) {
qWarning ( ) < < this - > m_path < < " empty, skipping " ;
} else {
qWarning ( ) < < this - > m_path < < " file too short? " ;
}
2018-03-23 19:24:29 +00:00
break ;
}
2019-05-15 19:16:14 +00:00
2019-05-15 02:49:41 +00:00
unsigned char * header = ( unsigned char * ) this - > m_header . data ( ) ;
this - > fileVersion = header [ 0 ] ; // Correlates to DataFileVersion in PROP[erties].TXT, only 2 or 3 has ever been observed
this - > blockSize = ( header [ 2 ] < < 8 ) | header [ 1 ] ;
this - > htype = header [ 3 ] ; // 00 = normal, 01=waveform
this - > family = header [ 4 ] ;
this - > familyVersion = header [ 5 ] ;
this - > ext = header [ 6 ] ;
this - > sessionid = ( header [ 10 ] < < 24 ) | ( header [ 9 ] < < 16 ) | ( header [ 8 ] < < 8 ) | header [ 7 ] ;
this - > timestamp = ( header [ 14 ] < < 24 ) | ( header [ 13 ] < < 16 ) | ( header [ 12 ] < < 8 ) | header [ 11 ] ;
2018-03-23 19:24:29 +00:00
2019-05-15 19:16:14 +00:00
// Do a few early sanity checks before any variable-length header data.
2019-05-15 02:49:41 +00:00
if ( this - > blockSize = = 0 ) {
qWarning ( ) < < this - > m_path < < " blocksize 0? " ;
2018-03-23 19:24:29 +00:00
break ;
2019-05-13 16:11:04 +00:00
}
2019-05-15 02:49:41 +00:00
if ( this - > fileVersion < 2 | | this - > fileVersion > 3 ) {
2019-05-15 21:41:37 +00:00
qWarning ( ) < < this - > m_path < < " @ " < < hex < < this - > m_filepos < < " Never seen PRS1 header version < 2 or > 3 before " ;
2018-03-23 19:24:29 +00:00
break ;
}
2019-05-18 23:17:55 +00:00
if ( this - > htype ! = PRS1_HTYPE_NORMAL & & this - > htype ! = PRS1_HTYPE_INTERVAL ) {
qWarning ( ) < < this - > m_path < < " unexpected htype: " < < this - > htype ;
2019-06-04 02:01:02 +00:00
break ;
2019-05-18 23:17:55 +00:00
}
2018-03-23 19:24:29 +00:00
2019-05-15 19:16:14 +00:00
// Read format-specific variable-length header data.
2019-05-15 16:32:39 +00:00
bool hdr_ok = false ;
2019-05-18 23:17:55 +00:00
if ( this - > htype ! = PRS1_HTYPE_INTERVAL ) { // Not just waveforms: the 1160P uses this for its .002 events file.
// Not a waveform/interval chunk
2019-05-15 16:32:39 +00:00
switch ( this - > fileVersion ) {
case 2 :
hdr_ok = ReadNormalHeaderV2 ( f ) ;
2019-05-15 00:47:00 +00:00
break ;
2019-05-15 16:32:39 +00:00
case 3 :
hdr_ok = ReadNormalHeaderV3 ( f ) ;
break ;
default :
//hdr_ok remains false, warning is above
2018-04-07 19:20:04 +00:00
break ;
2019-05-14 23:57:01 +00:00
}
2019-05-18 23:17:55 +00:00
} else {
// Waveform/interval chunk
2019-05-15 16:32:39 +00:00
hdr_ok = ReadWaveformHeader ( f ) ;
}
if ( ! hdr_ok ) {
break ;
2019-05-15 00:47:00 +00:00
}
2019-05-15 01:49:43 +00:00
// The 8bit checksum comes at the end.
2019-05-15 00:47:00 +00:00
QByteArray checksum = f . read ( 1 ) ;
if ( checksum . size ( ) < 1 ) {
2019-05-15 02:49:41 +00:00
qWarning ( ) < < this - > m_path < < " read error header checksum " ;
2019-05-15 00:47:00 +00:00
break ;
}
2019-05-15 02:49:41 +00:00
this - > storedChecksum = checksum . data ( ) [ 0 ] ;
2018-03-23 19:24:29 +00:00
2019-05-15 01:49:43 +00:00
// Calculate 8bit additive header checksum.
2019-05-15 02:49:41 +00:00
header = ( unsigned char * ) this - > m_header . data ( ) ; // important because its memory location could move
int header_size = this - > m_header . size ( ) ;
2019-05-15 00:47:00 +00:00
quint8 achk = 0 ;
for ( int i = 0 ; i < header_size ; i + + ) {
achk + = header [ i ] ;
}
2019-05-15 02:49:41 +00:00
this - > calcChecksum = achk ;
2019-05-15 00:47:00 +00:00
2019-05-15 01:49:43 +00:00
// Append the stored checksum to the raw data *after* calculating the checksum on the preceding data.
2019-05-15 02:49:41 +00:00
this - > m_header . append ( checksum ) ;
2018-03-23 19:24:29 +00:00
2019-05-15 12:36:31 +00:00
ok = true ;
} while ( false ) ;
2018-03-23 19:24:29 +00:00
2019-05-15 12:36:31 +00:00
return ok ;
}
2019-05-15 16:32:39 +00:00
bool PRS1DataChunk : : ReadNormalHeaderV2 ( QFile & /*f*/ )
{
this - > m_headerblock = QByteArray ( ) ;
return true ; // always OK
}
bool PRS1DataChunk : : ReadNormalHeaderV3 ( QFile & f )
{
bool ok = false ;
unsigned char * header ;
QByteArray headerB2 ;
// This is a new machine, byte 15 is header data block length
// followed by variable, data byte pairs
do {
QByteArray extra = f . read ( 1 ) ;
if ( extra . size ( ) < 1 ) {
qWarning ( ) < < this - > m_path < < " read error extended header " ;
break ;
}
this - > m_header . append ( extra ) ;
header = ( unsigned char * ) this - > m_header . data ( ) ;
int hdb_len = header [ 15 ] ;
int hdb_size = hdb_len * 2 ;
headerB2 = f . read ( hdb_size ) ;
if ( headerB2 . size ( ) ! = hdb_size ) {
qWarning ( ) < < this - > m_path < < " read error in extended header " ;
break ;
}
this - > m_headerblock = headerB2 ;
this - > m_header . append ( headerB2 ) ;
header = ( unsigned char * ) this - > m_header . data ( ) ;
const unsigned char * hd = ( unsigned char * ) headerB2 . constData ( ) ;
int pos = 0 ;
int recs = header [ 15 ] ;
for ( int i = 0 ; i < recs ; i + + ) {
this - > hblock [ hd [ pos ] ] = hd [ pos + 1 ] ;
pos + = 2 ;
}
ok = true ;
} while ( false ) ;
return ok ;
}
2019-05-15 14:26:31 +00:00
bool PRS1DataChunk : : ReadWaveformHeader ( QFile & f )
{
bool ok = false ;
unsigned char * header ;
do {
2019-05-18 23:17:55 +00:00
// Read the fixed-length waveform header.
QByteArray extra = f . read ( 4 ) ;
if ( extra . size ( ) ! = 4 ) {
2019-05-15 14:26:31 +00:00
qWarning ( ) < < this - > m_path < < " read error in waveform header " ;
break ;
}
this - > m_header . append ( extra ) ;
header = ( unsigned char * ) this - > m_header . data ( ) ;
2019-05-18 23:17:55 +00:00
// Parse the fixed-length portion.
this - > interval_count = header [ 0x0f ] | header [ 0x10 ] < < 8 ;
this - > interval_seconds = header [ 0x11 ] ; // not always 1 after all
this - > duration = this - > interval_count * this - > interval_seconds ; // ??? the last entry doesn't always seem to be a full interval?
quint8 wvfm_signals = header [ 0x12 ] ;
2019-05-15 14:26:31 +00:00
2019-05-18 23:17:55 +00:00
// Read the variable-length data + trailing byte.
2019-05-15 14:26:31 +00:00
int ws_size = ( this - > fileVersion = = 3 ) ? 4 : 3 ;
2019-05-18 23:17:55 +00:00
int sbsize = wvfm_signals * ws_size + 1 ;
2019-05-15 14:26:31 +00:00
extra = f . read ( sbsize ) ;
if ( extra . size ( ) ! = sbsize ) {
qWarning ( ) < < this - > m_path < < " read error in waveform header 2 " ;
break ;
}
this - > m_header . append ( extra ) ;
header = ( unsigned char * ) this - > m_header . data ( ) ;
2019-05-18 23:17:55 +00:00
// Parse the variable-length waveform information.
2019-06-04 02:01:02 +00:00
// TODO: move these checks into the parser, after the header checksum has been verified
2020-01-28 14:34:02 +00:00
// For now just skip them for the one known sample with a bad checksum.
if ( this - > sessionid = = 268962649 ) return true ;
2019-05-18 23:17:55 +00:00
int pos = 0x13 ;
2019-05-15 14:26:31 +00:00
for ( int i = 0 ; i < wvfm_signals ; + + i ) {
2019-05-18 23:17:55 +00:00
quint8 kind = header [ pos ] ;
2020-01-28 14:34:02 +00:00
CHECK_VALUE ( kind , i ) ; // always seems to range from 0...wvfm_signals-1, alert if not
2019-05-18 23:17:55 +00:00
quint16 interleave = header [ pos + 1 ] | header [ pos + 2 ] < < 8 ; // samples per interval
2019-05-15 14:26:31 +00:00
if ( this - > fileVersion = = 2 ) {
2019-05-18 23:17:55 +00:00
this - > waveformInfo . push_back ( PRS1Waveform ( interleave , kind ) ) ;
pos + = 3 ;
2019-05-15 14:26:31 +00:00
} else if ( this - > fileVersion = = 3 ) {
2019-05-18 23:17:55 +00:00
int always_8 = header [ pos + 3 ] ; // sample size in bits?
2020-01-28 14:34:02 +00:00
CHECK_VALUE ( always_8 , 8 ) ;
2019-05-18 23:17:55 +00:00
this - > waveformInfo . push_back ( PRS1Waveform ( interleave , kind ) ) ;
pos + = 4 ;
2019-05-15 14:26:31 +00:00
}
}
2019-05-18 23:17:55 +00:00
// And the trailing byte, whatever it is.
int always_0 = header [ pos ] ;
2020-01-28 14:34:02 +00:00
CHECK_VALUE ( always_0 , 0 ) ;
2019-05-18 23:17:55 +00:00
2019-05-15 14:26:31 +00:00
ok = true ;
} while ( false ) ;
return ok ;
}
2019-05-15 12:36:31 +00:00
bool PRS1DataChunk : : ReadData ( QFile & f )
{
bool ok = false ;
do {
2018-03-23 19:24:29 +00:00
// Read data block
2019-05-15 02:49:41 +00:00
int data_size = this - > blockSize - this - > m_header . size ( ) ;
2019-05-15 19:16:14 +00:00
if ( data_size < 0 ) {
qWarning ( ) < < this - > m_path < < " chunk size smaller than header " ;
break ;
}
2019-05-15 02:49:41 +00:00
this - > m_data = f . read ( data_size ) ;
if ( this - > m_data . size ( ) < data_size ) {
qWarning ( ) < < this - > m_path < < " less data in file than specified in header " ;
2018-03-23 19:24:29 +00:00
break ;
}
2019-05-15 19:16:14 +00:00
// Extract the stored CRC from the data buffer and calculate the current CRC.
2019-05-15 02:49:41 +00:00
if ( this - > fileVersion = = 3 ) {
2019-05-15 19:16:14 +00:00
// The last 4 bytes contain a CRC32 checksum of the data.
if ( ! ExtractStoredCrc ( 4 ) ) {
break ;
}
2019-05-15 21:41:37 +00:00
this - > calcCrc = CRC32wchar ( ( unsigned char * ) this - > m_data . data ( ) , this - > m_data . size ( ) ) ;
2018-03-23 19:24:29 +00:00
} else {
2019-05-15 19:16:14 +00:00
// The last 2 bytes contain a CRC16 checksum of the data.
if ( ! ExtractStoredCrc ( 2 ) ) {
break ;
2018-03-23 19:24:29 +00:00
}
2019-05-15 19:16:14 +00:00
this - > calcCrc = CRC16 ( ( unsigned char * ) this - > m_data . data ( ) , this - > m_data . size ( ) ) ;
2018-03-23 19:24:29 +00:00
}
2019-05-14 22:57:04 +00:00
2019-05-15 02:49:41 +00:00
ok = true ;
2019-05-14 22:57:04 +00:00
} while ( false ) ;
2018-03-23 19:24:29 +00:00
2019-05-15 02:49:41 +00:00
return ok ;
2018-03-23 19:24:29 +00:00
}
2019-05-15 19:16:14 +00:00
bool PRS1DataChunk : : ExtractStoredCrc ( int size )
{
// Make sure there's enough data for the CRC.
int offset = this - > m_data . size ( ) - size ;
if ( offset < 0 ) {
qWarning ( ) < < this - > m_path < < " chunk truncated " ;
return false ;
}
// Read the last 16- or 32-bit little-endian integer.
quint32 storedCrc = 0 ;
unsigned char * data = ( unsigned char * ) this - > m_data . data ( ) ;
for ( int i = 0 ; i < size ; i + + ) {
storedCrc | = data [ offset + i ] < < ( 8 * i ) ;
}
this - > storedCrc = storedCrc ;
// Drop the CRC from the data.
this - > m_data . chop ( size ) ;
return true ;
}
2019-05-15 02:49:41 +00:00
2018-03-23 19:24:29 +00:00
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 " ) ) ) ;
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 " ) ) ;
2019-09-20 16:59:14 +00:00
chan - > addOption ( FLEX_AVAPS , QObject : : tr ( " AVAPS " ) ) ;
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-05-17 05:04:40 +00:00
2014-08-03 13:00:13 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , QObject : : tr ( " x1 " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " x2 " ) ) ;
chan - > addOption ( 3 , QObject : : tr ( " x3 " ) ) ;
2014-08-04 19:57:48 +00:00
chan - > addOption ( 4 , QObject : : tr ( " x4 " ) ) ;
chan - > addOption ( 5 , QObject : : tr ( " x5 " ) ) ;
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 ) ;
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-01-12 00:14:01 +00:00
chan - > addOption ( 0 , QObject : : tr ( " Fixed (Classic) " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " Adaptive (System One) " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " Heated Tube " ) ) ;
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 ) ;
chan - > addOption ( 1 , QObject : : tr ( " 1 " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " 2 " ) ) ;
chan - > addOption ( 3 , QObject : : tr ( " 3 " ) ) ;
chan - > addOption ( 4 , QObject : : tr ( " 4 " ) ) ;
chan - > addOption ( 5 , QObject : : tr ( " 5 " ) ) ;
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-01-13 01:13:21 +00:00
QObject : : tr ( " Humid. Lvl " ) ,
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-01-12 00:14:01 +00:00
chan - > addOption ( 1 , QObject : : tr ( " 1 " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " 2 " ) ) ;
chan - > addOption ( 3 , QObject : : tr ( " 3 " ) ) ;
chan - > addOption ( 4 , QObject : : tr ( " 4 " ) ) ;
chan - > addOption ( 5 , QObject : : tr ( " 5 " ) ) ;
2014-08-04 19:57:48 +00:00
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 ) ;
chan - > addOption ( 1 , QObject : : tr ( " x1 " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " x2 " ) ) ;
chan - > addOption ( 3 , QObject : : tr ( " x3 " ) ) ;
chan - > addOption ( 4 , QObject : : tr ( " x4 " ) ) ;
chan - > addOption ( 5 , QObject : : tr ( " x5 " ) ) ;
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 " ) ,
QObject : : tr ( " Hose Diameter " ) ,
" " , 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. " ) ,
QObject : : tr ( " Mask Resist. 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 " ) ,
QObject : : tr ( " A few breaths automatically starts machine " ) ,
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 " ) ,
QObject : : tr ( " Machine automatically switches off " ) ,
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 " ) ,
QObject : : tr ( " Whether or not machine allows Mask checking. " ) ,
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 " ) ,
2020-03-23 16:59:06 +00:00
QObject : : tr ( " Whether or not machine 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 " ) ) ;
// TODO: is the below useful?
2014-08-11 01:44:25 +00:00
// <channel id="0xe10e" class="setting" scope="!session" name="PRS1Mode" details="PAP Mode" label="PAP Mode" type="integer" link="0x1200">
2014-08-04 19:57:48 +00:00
// <Option id="0" value="CPAP"/>
// <Option id="1" value="Auto"/>
// <Option id="2" value="BIPAP"/>
// <Option id="3" value="AutoSV"/>
// </channel>
2019-10-29 18:06:57 +00:00
channel . add ( GRP_CPAP , new Channel ( PRS1_0E = 0x1157 , SPAN , MT_CPAP , SESSION ,
2019-12-28 04:08:59 +00:00
" PRS1_UNK " ,
QObject : : tr ( " PRS1 Unknown " ) ,
QObject : : tr ( " Unknown PRS1 span 0x0E " ) ,
2019-10-29 18:06:57 +00:00
" ?? " ,
2019-12-28 04:08:59 +00:00
STR_UNIT_Seconds ,
2019-10-29 18:06:57 +00:00
DEFAULT , QColor ( " #ffe8f0 " ) ) ) ;
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 " ) ,
QObject : : tr ( " A period during a session where the machine could not detect flow. " ) ,
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 " ) ) ) ;
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
}
2019-02-13 18:22:54 +00:00
/* Thanks to SleepyCPAP :)
2014-08-20 18:36:44 +00:00
CODE ERROR DESCRIPTION ERROR TYPE ERROR CATEGORY
1 SOFTWARE STOP STOP General Errors
2 Not Used General Errors
3 INT RAM REBOOT General Errors
4 NULL PTR REBOOT General Errors
5 DATA REBOOT General Errors
6 STATE MACHINE REBOOT General Errors
7 SOFTWARE REBOOT General Errors
8 - 9 Not Used General Errors
10 WDOG TEST RAM REBOOT Watchdog & Timer Errors
11 WDOG TEST REBOOT Watchdog & Timer Errors
12 BACKGROUND WDOG NO CARD REBOOT Watchdog & Timer Errors
13 BACKGROUND WDOG SD CARD REBOOT Watchdog & Timer Errors
14 WDOG LOWRES TIMER REBOOT Watchdog & Timer Errors
15 CYCLE HANDLER OVERRUN REBOOT Watchdog & Timer Errors
16 RASP RESTORE TIMEOUT CONTINUE Watchdog & Timer Errors
17 ONEMS HANDLER OVERRUN REBOOT Watchdog & Timer Errors
18 Not Used Watchdog & Timer Errors
19 WDOG TIMEOUT REBOOT Watchdog & Timer Errors
20 MOTOR SPINUP FLUX LOW REBOOT Motor / Blower Errors
21 MOTOR VBUS HIGH STOP Motor / Blower Errors
22 MOTOR FLUX MAGNITUDE REBOOT Motor / Blower Errors
23 MOTOR OVERSPEED REBOOT Motor / Blower Errors
24 MOTOR SPEED REVERSE REBOOT Motor / Blower Errors
25 MOTOR THERMISTOR OPEN CONTINUE Motor / Blower Errors
26 MOTOR THERMISTOR SHORTED CONTINUE Motor / Blower Errors
27 MOTOR RL NOCONVERGE STOP Motor / Blower Errors
28 NEGATIVE QUADRATURE VOLTAGE VECTOR REBOOT Motor / Blower Errors
29 VBUS GAIN ZERO : REBOOT Motor / Blower Errors
30 MOTOR SPINUP FLUX HIGH REBOOT Motor / Blower Errors
31 ( incorrect power supply - 60 series ) Motor / Blower Errors
32 - 39 Not Used Motor / Blower Errors
40 NVRAM REBOOT NVRAM Low Level Errors
41 STORAGE UNIT RAM REBOOT NVRAM Low Level Errors
42 UNABLE TO OBTAIN BUS REBOOT NVRAM Low Level Errors
43 NVRAM NO CALLBACK OCCURRED REBOOT NVRAM Low Level Errors
44 NV BUFFER NULL REBOOT NVRAM Low Level Errors
45 NV CALLBACK NULL REBOOT NVRAM Low Level Errors
46 NV ZERO LENGTH REBOOT NVRAM Low Level Errors
47 NVRAM INVALID BYTES XFRRED REBOOT NVRAM Low Level Errors
48 - 49 Not Used NVRAM Low Level Errors
50 DAILY VALUES CORRUPT LOG ONLY NVRAM Unit Related Errors
51 CORRUPT COMPLIANCE LOG CONTINUE NVRAM Unit Related Errors
52 CORRUPT COMPLIANCE CB CONTINUE NVRAM Unit Related Errors
53 COMP LOG SEM TIMEOUT CONTINUE NVRAM Unit Related Errors
54 COMPLOG REQS OVERFLOW REBOOT NVRAM Unit Related Errors
55 THERAPY QUEUE FULL CONTINUE NVRAM Unit Related Errors
56 COMPLOG PACKET STATUS REBOOT NVRAM Unit Related Errors
57 SESS OBS QUEUE OVF REBOOT NVRAM Unit Related Errors
58 SESS OBS NO CALLBACK REBOOT NVRAM Unit Related Errors
59 Not Used NVRAM Unit Related Errors
60 UNSUPPORTED HARDWARE REBOOT General Hardware Errors
61 PLL UNLOCKED REBOOT General Hardware Errors
62 STUCK RAMP KEY CONTINUE General Hardware Errors
63 STUCK KNOB KEY CONTINUE General Hardware Errors
64 DSP OVERTIME PWM REBOOT General Hardware Errors
65 STUCK ENCODER A CONTINUE General Hardware Errors
66 STUCK ENCODER B CONTINUE General Hardware Errors
67 - 69 Not Used General Hardware Errors
70 PRESSURE SENSOR ABSENT STOP Pressure Sensor Errors
71 Not Used Pressure Sensor Errors
72 PSENS UNABLE TO OBTAIN BUS REBOOT Pressure Sensor Errors
73 SENSOR PRESS OFFSET STOP STOP Pressure Sensor Errors
74 - 79 Not Used Pressure Sensor Errors
80 UNABLE TO INIT FLOW SENSOR REBOOT Flow Sensor Errors
81 FLOW SENSOR TABLE CONTINUE Flow Sensor Errors
82 FLOW SENSOR OFFSET CONTINUE Flow Sensor Errors
83 FSENS UNABLE TO OBTAIN BUS REBOOT / 2 nd failure = STOP Flow Sensor Errors
84 FLOW SENSOR STOP STOP Flow Sensor Errors
85 FLOW SENSOR OCCLUDED CONTINUE Flow Sensor Errors
86 FLOW SENSOR ABSENT CONTINUE Flow Sensor Errors
87 FLOW SENSOR BUS CONTINUE Flow Sensor Errors
88 - 89 Not Used Flow Sensor Errors
90 OTP NOT CONFIGURED STOP OTP & RTC Errors
91 OTP INCORRECTLY CONFIGURED STOP OTP & RTC Errors
92 Not Used OTP & RTC Errors
93 RTC VALUE CONTINUE OTP & RTC Errors
94 RTC STOPPED CONTINUE OTP & RTC Errors
95 - 99 Not Used OTP & RTC Errors
100 HUMID NO HEAT CONTINUE Humidifier Errors
101 HUMID TEMP MAX STOP Humidifier Errors
102 THERMISTOR HIGH CONTINUE Humidifier Errors
103 THERMISTOR LOW CONTINUE Humidifier Errors
104 HUMID AMBIENT OFF CONTINUE Humidifier Errors
105 HUMID AMBIENT COMM CONTINUE Humidifier Errors
106 - 109 Not Used Humidifier Errors
110 STACK REBOOT Stack & Exception Handler Errors
111 EXCEPTION STACK OVERFLOW REBOOT Stack & Exception Handler Errors
112 EXCEPTION STACK RESERVE LOG ONLY Stack & Exception Handler Errors
113 EXCEPTION STACK UNDERFLOW REBOOT Stack & Exception Handler Errors
114 FIQ STACK OVERFLOW REBOOT Stack & Exception Handler Errors
115 FIQ STACK RESERVE LOG ONLY Stack & Exception Handler Errors
116 FIQ STACK UNDERFLOW REBOOT Stack & Exception Handler Errors
117 IRQ STACK OVERFLOW REBOOT Stack & Exception Handler Errors
118 IRQ STACK RESERVE LOG ONLY Stack & Exception Handler Errors
119 IRQ STACK UNDERFLOW REBOOT Stack & Exception Handler Errors
120 SVC STACK OVERFLOW REBOOT Stack & Exception Handler Errors
121 SVC STACK RESERVE LOG ONLY Stack & Exception Handler Errors
122 SVC STACK UNDERFLOW REBOOT Stack & Exception Handler Errors
123 DATA ABORT EXCEPTION REBOOT Stack & Exception Handler Errors
124 PREFETCH EXCEPTION REBOOT Stack & Exception Handler Errors
125 ILLEGAL INSTRUCTION EXCEPTION REBOOT Stack & Exception Handler Errors
126 SWI ABORT EXCEPTION REBOOT Stack & Exception Handler Errors
*/
2012-01-19 15:18:34 +00:00