2019-07-10 16:33:00 +00:00
/* SleepLib ResMed Loader Implementation
2014-04-09 21:01:57 +00:00
*
2024-01-13 20:27:48 +00:00
* Copyright ( c ) 2019 - 2024 The OSCAR Team
2024-02-01 00:14:19 +00:00
* Copyright ( c ) 2011 - 2018 Mark Watkins
2014-04-09 21:01:57 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
2018-06-04 20:48:38 +00:00
* License . See the file COPYING in the main directory of the source code
* for more details . */
2011-07-30 00:36:31 +00:00
2023-03-18 23:37:13 +00:00
# define TEST_MACROS_ENABLEDoff
2023-02-18 13:58:47 +00:00
# include <test_macros.h>
2011-07-30 00:36:31 +00:00
# include <QApplication>
2011-06-28 02:21:38 +00:00
# include <QString>
# include <QDateTime>
# include <QDir>
# include <QFile>
# include <QMessageBox>
2014-07-27 16:35:49 +00:00
# include <QTextStream>
2011-07-01 10:10:44 +00:00
# include <QDebug>
2015-08-02 05:53:09 +00:00
# include <QStringList>
2021-06-07 02:57:52 +00:00
# include <QJsonDocument>
# include <QJsonObject>
2011-07-27 09:21:53 +00:00
# include <cmath>
2011-06-28 02:21:38 +00:00
# include "SleepLib/session.h"
2011-11-28 01:39:28 +00:00
# include "SleepLib/calcs.h"
2011-06-28 02:21:38 +00:00
2019-11-17 21:41:47 +00:00
# include "SleepLib/loader_plugins/resmed_loader.h"
# include "SleepLib/loader_plugins/resmed_EDFinfo.h"
2012-01-05 06:54:07 +00:00
# ifdef DEBUG_EFFICIENCY
2019-12-24 02:44:10 +00:00
# include <QElapsedTimer> // only available in 4.8 and later
2012-01-05 06:54:07 +00:00
# endif
2014-05-04 18:02:41 +00:00
2023-02-18 13:58:47 +00:00
# if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
# define QTCOMBINE insert
//idmap.insert(hash);
# else
# define QTCOMBINE unite
//idmap.unite(hash);
# endif
2016-03-08 13:55:29 +00:00
ChannelID RMS9_EPR , RMS9_EPRLevel , RMS9_Mode , RMS9_SmartStart , RMS9_HumidStatus , RMS9_HumidLevel ,
2021-08-17 15:32:39 +00:00
RMS9_PtAccess , RMS9_Mask , RMS9_ABFilter , RMS9_ClimateControl , RMS9_TubeType , RMAS11_SmartStop ,
RMS9_Temp , RMS9_TempEnable , RMS9_RampEnable , RMAS1x_Comfort , RMAS11_PtView ;
2014-08-03 13:00:13 +00:00
2022-04-22 16:33:32 +00:00
ChannelID RMAS1x_EasyBreathe , RMAS1x_RiseEnable , RMAS1x_RiseTime , RMAS1x_Cycle , RMAS1x_Trigger , RMAS1x_TiMax , RMAS1x_TiMin ;
2022-03-09 21:42:39 +00:00
2014-09-17 17:20:01 +00:00
const QString STR_ResMed_AirSense10 = " AirSense 10 " ;
2021-07-08 18:27:22 +00:00
const QString STR_ResMed_AirSense11 = " AirSense 11 " ;
const QString STR_ResMed_AirCurve10 = " AirCurve 10 " ;
const QString STR_ResMed_AirCurve11 = " AirCurve 11 " ;
2014-09-17 17:20:01 +00:00
const QString STR_ResMed_S9 = " S9 " ;
2019-04-05 18:00:37 +00:00
const QString STR_UnknownModel = " Resmed ??? " ;
2014-08-03 13:00:13 +00:00
2021-08-01 03:30:24 +00:00
// TODO: See the PRSLoader::LogUnexpectedMessage TODO about generalizing this for other loaders.
void ResmedLoader : : LogUnexpectedMessage ( const QString & message )
{
m_importMutex . lock ( ) ;
m_unexpectedMessages + = message ;
m_importMutex . unlock ( ) ;
}
2024-09-20 00:08:43 +00:00
static const QVector < int > AS11TestedModels { 39463 , 39420 , 39421 , 39423 , 39483 , 39485 , 39517 , 39520 , 39494 , 39491 , 0 } ;
2021-08-01 03:30:24 +00:00
2019-12-24 02:44:10 +00:00
ResmedLoader : : ResmedLoader ( ) {
# ifndef UNITTEST_MODE
const QString RMS9_ICON = " :/icons/rms9.png " ;
const QString RM10_ICON = " :/icons/airsense10.png " ;
const QString RM10C_ICON = " :/icons/aircurve.png " ;
2014-05-04 18:02:41 +00:00
2019-12-24 02:44:10 +00:00
m_pixmaps [ STR_ResMed_S9 ] = QPixmap ( RMS9_ICON ) ;
m_pixmap_paths [ STR_ResMed_S9 ] = RMS9_ICON ;
m_pixmaps [ STR_ResMed_AirSense10 ] = QPixmap ( RM10_ICON ) ;
m_pixmap_paths [ STR_ResMed_AirSense10 ] = RM10_ICON ;
m_pixmaps [ STR_ResMed_AirCurve10 ] = QPixmap ( RM10C_ICON ) ;
m_pixmap_paths [ STR_ResMed_AirCurve10 ] = RM10C_ICON ;
# endif
m_type = MT_CPAP ;
2011-06-30 04:55:20 +00:00
2022-01-05 20:35:35 +00:00
# ifdef DEBUG_EFFICIENCY
2019-12-24 02:44:10 +00:00
timeInTimeDelta = timeInLoadBRP = timeInLoadPLD = timeInLoadEVE = 0 ;
timeInLoadCSL = timeInLoadSAD = timeInEDFInfo = timeInEDFOpen = timeInAddWaveform = 0 ;
2022-01-05 20:35:35 +00:00
# endif
2013-09-14 23:32:14 +00:00
2020-02-10 21:04:03 +00:00
saveCallback = SaveSession ;
2019-12-24 02:44:10 +00:00
}
ResmedLoader : : ~ ResmedLoader ( ) { }
bool resmed_initialized = false ;
void ResmedLoader : : Register ( )
2014-05-04 18:02:41 +00:00
{
2019-12-24 02:44:10 +00:00
if ( resmed_initialized )
return ;
2014-05-04 18:02:41 +00:00
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Registering ResmedLoader " ;
RegisterLoader ( new ResmedLoader ( ) ) ;
resmed_initialized = true ;
2014-05-04 18:02:41 +00:00
}
2019-12-24 02:44:10 +00:00
void setupResMedTranslationMap ( ) ; // forward
void ResmedLoader : : initChannels ( )
2014-05-18 17:06:58 +00:00
{
2019-12-24 02:44:10 +00:00
using namespace schema ;
2020-03-31 00:38:22 +00:00
// Channel(ChannelID id, ChanType type, MachineType machtype, ScopeType scope, QString code, QString fullname,
// QString description, QString label, QString unit, DataType datatype = DEFAULT, QColor = Qt::black, int link = 0);
2019-12-24 02:44:10 +00:00
Channel * chan = new Channel ( RMS9_Mode = 0xe203 , SETTING , MT_CPAP , SESSION ,
" RMS9_Mode " , QObject : : tr ( " Mode " ) , QObject : : tr ( " CPAP Mode " ) , QObject : : tr ( " Mode " ) , " " , LOOKUP , Qt : : green ) ;
2018-05-05 21:58:11 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan ) ;
2018-05-07 01:30:42 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , QObject : : tr ( " CPAP " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " APAP " ) ) ;
2024-03-05 18:48:13 +00:00
chan - > addOption ( 2 , QObject : : tr ( " BiLevel-T " ) ) ;
chan - > addOption ( 3 , QObject : : tr ( " BiLevel-S " ) ) ;
chan - > addOption ( 4 , QObject : : tr ( " BiLevel-S/T " ) ) ;
chan - > addOption ( 5 , QObject : : tr ( " BiLevel-T " ) ) ;
2019-12-24 02:44:10 +00:00
chan - > addOption ( 6 , QObject : : tr ( " VPAPauto " ) ) ;
chan - > addOption ( 7 , QObject : : tr ( " ASV " ) ) ;
chan - > addOption ( 8 , QObject : : tr ( " ASVAuto " ) ) ;
2021-08-22 19:21:12 +00:00
chan - > addOption ( 9 , QObject : : tr ( " iVAPS " ) ) ;
2021-10-07 23:32:19 +00:00
chan - > addOption ( 10 , QObject : : tr ( " PAC " ) ) ;
2019-12-24 02:44:10 +00:00
chan - > addOption ( 11 , QObject : : tr ( " Auto for Her " ) ) ;
2021-10-28 00:26:18 +00:00
chan - > addOption ( 16 , QObject : : tr ( " Unknown " ) ) ; // out of bounds of edf signal
2018-05-07 18:42:23 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_EPR = 0xe201 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" EPR " , QObject : : tr ( " EPR " ) , QObject : : tr ( " ResMed Exhale Pressure Relief " ) , QObject : : tr ( " EPR " ) , " " , LOOKUP , Qt : : green ) ) ;
2018-05-07 18:42:23 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , QObject : : tr ( " Ramp Only " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " Full Time " ) ) ;
chan - > addOption ( 3 , QObject : : tr ( " Patient??? " ) ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_EPRLevel = 0xe202 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" EPRLevel " , QObject : : tr ( " EPR Level " ) , QObject : : tr ( " Exhale Pressure Relief Level " ) , QObject : : tr ( " EPR Level " ) , STR_UNIT_CMH2O , LOOKUP , Qt : : blue ) ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
// RMS9_SmartStart, RMS9_HumidStatus, RMS9_HumidLevel,
// RMS9_PtAccess, RMS9_Mask, RMS9_ABFilter, RMS9_ClimateControl, RMS9_TubeType,
// RMS9_Temp, RMS9_TempEnable;
2018-04-27 04:29:03 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_SmartStart = 0xe204 , SETTING , MT_CPAP , SESSION ,
2022-02-27 16:50:10 +00:00
" RMS9_SmartStart " , QObject : : tr ( " SmartStart " ) , QObject : : tr ( " Device auto starts by breathing " ) , QObject : : tr ( " Smart Start " ) , " " , LOOKUP , Qt : : black ) ) ;
2018-04-27 04:29:03 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_HumidStatus = 0xe205 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_HumidStat " , QObject : : tr ( " Humid. Status " ) , QObject : : tr ( " Humidifier Enabled Status " ) , QObject : : tr ( " Humidifier Status " ) , " " , LOOKUP , Qt : : black ) ) ;
2018-05-07 18:42:23 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_HumidLevel = 0xe206 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_HumidLevel " , QObject : : tr ( " Humid. Level " ) , QObject : : tr ( " Humidity Level " ) , QObject : : tr ( " Humidity Level " ) , " " , LOOKUP , Qt : : black ) ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2020-03-31 00:38:22 +00:00
// chan->addOption(1, "1");
// chan->addOption(2, "2");
// chan->addOption(3, "3");
// chan->addOption(4, "4");
// chan->addOption(5, "5");
// chan->addOption(6, "6");
// chan->addOption(7, "7");
// chan->addOption(8, "8");
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_Temp = 0xe207 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_Temp " , QObject : : tr ( " Temperature " ) , QObject : : tr ( " ClimateLine Temperature " ) , QObject : : tr ( " Temperature " ) , " ºC " , INTEGER , Qt : : black ) ) ;
2014-05-20 11:51:47 +00:00
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_TempEnable = 0xe208 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_TempEnable " , QObject : : tr ( " Temp. Enable " ) , QObject : : tr ( " ClimateLine Temperature Enable " ) , QObject : : tr ( " Temperature Enable " ) , " " , LOOKUP , Qt : : black ) ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2021-08-22 19:21:12 +00:00
chan - > addOption ( 1 , STR_TR_On ) ;
chan - > addOption ( 2 , STR_TR_Auto ) ;
2014-08-26 07:55:01 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_ABFilter = 0xe209 , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_ABFilter " , QObject : : tr ( " AB Filter " ) , QObject : : tr ( " Antibacterial Filter " ) , QObject : : tr ( " Antibacterial Filter " ) , " " , LOOKUP , Qt : : black ) ) ;
2014-08-26 07:55:01 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_No ) ;
chan - > addOption ( 1 , STR_TR_Yes ) ;
2014-08-26 07:55:01 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_PtAccess = 0xe20A , SETTING , MT_CPAP , SESSION ,
2020-09-01 05:02:32 +00:00
" RMS9_PTAccess " , QObject : : tr ( " Pt. Access " ) , QObject : : tr ( " Essentials " ) , QObject : : tr ( " Essentials " ) , " " , LOOKUP , Qt : : black ) ) ;
2014-08-26 07:55:01 +00:00
2020-09-01 05:02:32 +00:00
chan - > addOption ( 0 , QObject : : tr ( " Plus " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " On " ) ) ;
2015-09-21 01:30:23 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_ClimateControl = 0xe20B , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_ClimateControl " , QObject : : tr ( " Climate Control " ) , QObject : : tr ( " Climate Control " ) , QObject : : tr ( " Climate Control " ) , " " , LOOKUP , Qt : : black ) ) ;
2014-08-26 07:55:01 +00:00
2021-08-22 19:21:12 +00:00
chan - > addOption ( 0 , STR_TR_Auto ) ;
2019-12-24 02:44:10 +00:00
chan - > addOption ( 1 , QObject : : tr ( " Manual " ) ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_Mask = 0xe20C , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_Mask " , QObject : : tr ( " Mask " ) , QObject : : tr ( " ResMed Mask Setting " ) , QObject : : tr ( " Mask " ) , " " , LOOKUP , Qt : : black ) ) ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , QObject : : tr ( " Pillows " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " Full Face " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " Nasal " ) ) ;
2021-08-22 19:21:12 +00:00
chan - > addOption ( 3 , QObject : : tr ( " Unknown " ) ) ;
2014-09-17 17:20:01 +00:00
2019-12-24 02:44:10 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMS9_RampEnable = 0xe20D , SETTING , MT_CPAP , SESSION ,
2020-03-31 00:38:22 +00:00
" RMS9_RampEnable " , QObject : : tr ( " Ramp " ) , QObject : : tr ( " Ramp Enable " ) , QObject : : tr ( " Ramp " ) , " " , LOOKUP , Qt : : black ) ) ;
2018-05-06 16:59:50 +00:00
2019-12-24 02:44:10 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2021-08-22 19:21:12 +00:00
chan - > addOption ( 2 , STR_TR_Auto ) ;
2018-05-06 16:59:50 +00:00
2021-07-08 18:27:22 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_Comfort = 0xe20E , SETTING , MT_CPAP , SESSION ,
2021-08-24 23:56:59 +00:00
" RMAS1x_Comfort " , QObject : : tr ( " Response " ) , QObject : : tr ( " Response " ) , QObject : : tr ( " Response " ) , " " , LOOKUP , Qt : : black ) ) ;
2021-07-08 18:27:22 +00:00
2021-09-07 12:51:00 +00:00
chan - > addOption ( 0 , QObject : : tr ( " Standard " ) ) ; // This must be verified
chan - > addOption ( 1 , QObject : : tr ( " Soft " ) ) ;
2021-07-25 22:43:23 +00:00
2021-08-17 15:32:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS11_SmartStop = 0xe20F , SETTING , MT_CPAP , SESSION ,
2022-02-27 16:50:10 +00:00
" RMAS11_SmartStop " , QObject : : tr ( " SmartStop " ) , QObject : : tr ( " Device auto stops by breathing " ) , QObject : : tr ( " Smart Stop " ) , " " , LOOKUP , Qt : : black ) ) ;
2021-07-25 22:43:23 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2021-07-08 18:27:22 +00:00
chan - > addOption ( 1 , STR_TR_On ) ;
2021-08-17 15:32:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS11_PtView = 0xe210 , SETTING , MT_CPAP , SESSION ,
2021-08-24 23:56:59 +00:00
" RMAS11_PTView " , QObject : : tr ( " Patient View " ) , QObject : : tr ( " Patient View " ) , QObject : : tr ( " Patient View " ) , " " , LOOKUP , Qt : : black ) ) ;
2021-08-17 15:32:39 +00:00
2021-09-07 12:51:00 +00:00
chan - > addOption ( 0 , QObject : : tr ( " Advanced " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " Simple " ) ) ;
2021-08-17 15:32:39 +00:00
2022-05-01 19:56:44 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2022-03-09 21:42:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_RiseEnable = 0xe212 , SETTING , MT_CPAP , SESSION ,
" RMAS1x_RiseEnable " , QObject : : tr ( " RiseEnable " ) , QObject : : tr ( " RiseEnable " ) , QObject : : tr ( " RiseEnable " ) , " " , LOOKUP , Qt : : black ) ) ;
2022-05-01 19:56:44 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , " Enabled " ) ;
2022-03-09 21:42:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_RiseTime = 0xe213 , SETTING , MT_CPAP , SESSION ,
2022-12-17 16:40:59 +00:00
" RMAS1x_RiseTime " , QObject : : tr ( " RiseTime " ) , QObject : : tr ( " RiseTime " ) , QObject : : tr ( " RiseTime " ) , STR_UNIT_milliSeconds , INTEGER , Qt : : black ) ) ;
2022-03-09 21:42:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_Cycle = 0xe214 , SETTING , MT_CPAP , SESSION ,
" RMAS1x_Cycle " , QObject : : tr ( " Cycle " ) , QObject : : tr ( " Cycle " ) , QObject : : tr ( " Cycle " ) , " " , LOOKUP , Qt : : black ) ) ;
2022-05-01 19:56:44 +00:00
chan - > addOption ( 0 , " Very Low " ) ;
chan - > addOption ( 1 , " Low " ) ;
chan - > addOption ( 2 , " Med " ) ;
chan - > addOption ( 3 , " High " ) ;
chan - > addOption ( 4 , " Very High " ) ;
2022-03-09 21:42:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_Trigger = 0xe215 , SETTING , MT_CPAP , SESSION ,
" RMAS1x_Trigger " , QObject : : tr ( " Trigger " ) , QObject : : tr ( " Trigger " ) , QObject : : tr ( " Trigger " ) , " " , LOOKUP , Qt : : black ) ) ;
2022-05-01 19:56:44 +00:00
chan - > addOption ( 0 , " Very Low " ) ;
chan - > addOption ( 1 , " Low " ) ;
chan - > addOption ( 2 , " Med " ) ;
chan - > addOption ( 3 , " High " ) ;
chan - > addOption ( 4 , " Very High " ) ;
2022-03-09 21:42:39 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_TiMax = 0xe216 , SETTING , MT_CPAP , SESSION ,
2022-12-17 16:40:59 +00:00
" RMAS1x_TiMax " , QObject : : tr ( " TiMax " ) , QObject : : tr ( " TiMax " ) , QObject : : tr ( " TiMax " ) , STR_UNIT_Seconds , DOUBLE , Qt : : black ) ) ;
2022-03-09 21:42:39 +00:00
chan - > addOption ( 0 , " 0 " ) ;
channel . add ( GRP_CPAP , chan = new Channel ( RMAS1x_TiMin = 0xe217 , SETTING , MT_CPAP , SESSION ,
2022-12-17 16:40:59 +00:00
" RMAS1x_TiMin " , QObject : : tr ( " TiMin " ) , QObject : : tr ( " TiMin " ) , QObject : : tr ( " TiMin " ) , STR_UNIT_Seconds , DOUBLE , Qt : : black ) ) ;
2022-03-09 21:42:39 +00:00
chan - > addOption ( 0 , " 0 " ) ;
2019-12-24 02:44:10 +00:00
// Setup ResMeds signal name translation map
setupResMedTranslationMap ( ) ;
2011-06-28 02:21:38 +00:00
}
2019-07-22 21:01:47 +00:00
2019-12-24 02:44:10 +00:00
ChannelID ResmedLoader : : CPAPModeChannel ( ) { return RMS9_Mode ; }
ChannelID ResmedLoader : : PresReliefMode ( ) { return RMS9_EPR ; }
ChannelID ResmedLoader : : PresReliefLevel ( ) { return RMS9_EPRLevel ; }
2011-06-28 02:21:38 +00:00
2019-12-24 02:44:10 +00:00
QHash < ChannelID , QStringList > resmed_codes ;
const QString STR_ext_TGT = " tgt " ;
2021-06-07 02:57:52 +00:00
const QString STR_ext_JSON = " json " ;
2019-12-24 02:44:10 +00:00
const QString STR_ext_CRC = " crc " ;
2011-07-02 14:35:50 +00:00
2014-04-17 05:58:57 +00:00
const QString RMS9_STR_datalog = " DATALOG " ;
const QString RMS9_STR_idfile = " Identification. " ;
const QString RMS9_STR_strfile = " STR. " ;
2013-09-14 23:32:14 +00:00
2014-04-28 04:01:55 +00:00
bool ResmedLoader : : Detect ( const QString & givenpath )
{
2014-04-28 18:45:33 +00:00
QDir dir ( givenpath ) ;
2014-04-28 04:01:55 +00:00
2014-04-28 18:45:33 +00:00
if ( ! dir . exists ( ) ) {
return false ;
2014-04-28 04:01:55 +00:00
}
2014-04-28 18:45:33 +00:00
// ResMed drives contain a folder named "DATALOG".
if ( ! dir . exists ( RMS9_STR_datalog ) ) {
2014-04-28 04:01:55 +00:00
return false ;
}
2014-04-28 18:45:33 +00:00
// They also contain a file named "STR.edf".
if ( ! dir . exists ( " STR.edf " ) ) {
2014-04-28 04:01:55 +00:00
return false ;
2014-04-28 18:45:33 +00:00
}
2014-04-28 04:01:55 +00:00
return true ;
}
2013-09-14 23:32:14 +00:00
2019-12-24 02:44:10 +00:00
QHash < QString , QString > parseIdentLine ( const QString line , MachineInfo * info ) ; // forward
2021-06-07 02:57:52 +00:00
void scanProductObject ( const QJsonObject product , MachineInfo * info , QHash < QString , QString > * idmap ) ; // forward
2014-07-28 13:56:29 +00:00
MachineInfo ResmedLoader : : PeekInfo ( const QString & path )
{
2019-07-22 21:01:47 +00:00
if ( ! Detect ( path ) )
return MachineInfo ( ) ;
2014-07-28 13:56:29 +00:00
QFile f ( path + " / " + RMS9_STR_idfile + " tgt " ) ;
2021-09-12 17:45:24 +00:00
QFile j ( path + " / " + RMS9_STR_idfile + " json " ) ; // Check for AS11 file first, just in case
if ( j . exists ( ) ) { // somebody is reusing an SD card w/o re-formatting
2021-06-07 02:57:52 +00:00
if ( ! j . open ( QIODevice : : ReadOnly ) ) {
return MachineInfo ( ) ;
}
2021-09-12 17:45:24 +00:00
if ( f . exists ( ) ) {
qDebug ( ) < < " Old Ident.tgt file is ignored " ;
}
2021-06-07 02:57:52 +00:00
QByteArray identData = j . readAll ( ) ;
j . close ( ) ;
QJsonDocument identDoc ( QJsonDocument : : fromJson ( identData ) ) ;
QJsonObject identObj = identDoc . object ( ) ;
if ( identObj . contains ( " FlowGenerator " ) & & identObj [ " FlowGenerator " ] . isObject ( ) ) {
QJsonObject flow = identObj [ " FlowGenerator " ] . toObject ( ) ;
if ( flow . contains ( " IdentificationProfiles " ) & & flow [ " IdentificationProfiles " ] . isObject ( ) ) {
QJsonObject profiles = flow [ " IdentificationProfiles " ] . toObject ( ) ;
if ( profiles . contains ( " Product " ) & & profiles [ " Product " ] . isObject ( ) ) {
QJsonObject product = profiles [ " Product " ] . toObject ( ) ;
MachineInfo info = newInfo ( ) ;
scanProductObject ( product , & info , nullptr ) ;
return info ;
2021-10-28 03:23:38 +00:00
} else
qDebug ( ) < < " No Product in Profiles " ;
} else
qDebug ( ) < < " No IdentificationProfiles in FlowGenerator " ;
} else
qDebug ( ) < < " No FlowGenerator in Identification.json " ;
return MachineInfo ( ) ;
2014-07-28 13:56:29 +00:00
2021-06-07 02:57:52 +00:00
}
2021-09-12 17:45:24 +00:00
// Abort if this file is dodgy..
if ( f . exists ( ) ) {
if ( ! f . open ( QIODevice : : ReadOnly ) ) {
return MachineInfo ( ) ;
}
MachineInfo info = newInfo ( ) ;
// Parse # entries into idmap.
while ( ! f . atEnd ( ) ) {
QString line = f . readLine ( ) . trimmed ( ) ;
QHash < QString , QString > hash = parseIdentLine ( line , & info ) ;
}
return info ;
}
2021-06-07 02:57:52 +00:00
// neither filename exists, return empty info
return MachineInfo ( ) ;
2014-07-28 13:56:29 +00:00
}
2019-12-24 02:44:10 +00:00
long event_cnt = 0 ;
2014-07-27 16:35:49 +00:00
2021-06-07 02:57:52 +00:00
bool parseIdentFile ( QString path , MachineInfo * info , QHash < QString , QString > & idmap ) ; // forward
2020-05-17 23:24:31 +00:00
void backupSTRfiles ( const QString strpath , const QString importPath , const QString backupPath ,
2020-02-08 01:16:47 +00:00
MachineInfo & info , QMap < QDate , STRFile > & STRmap ) ; // forward
2020-05-17 23:24:31 +00:00
ResMedEDFInfo * fetchSTRandVerify ( QString filename , QString serialNumber ) ; // forward
2020-02-11 03:18:39 +00:00
2021-04-27 00:10:37 +00:00
int ResmedLoader : : OpenWithCallback ( const QString & dirpath , ResDaySaveCallback s ) // alternate for unit testing
2020-02-10 21:04:03 +00:00
{
ResDaySaveCallback origCallback = saveCallback ;
saveCallback = s ;
int value = Open ( dirpath ) ;
saveCallback = origCallback ;
return value ;
}
2019-12-24 02:44:10 +00:00
int ResmedLoader : : Open ( const QString & dirpath )
2014-09-01 04:49:05 +00:00
{
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Starting ResmedLoader::Open( with " < < dirpath < < " ) " ;
2019-12-24 02:44:10 +00:00
QString datalogPath ;
2022-02-27 16:50:10 +00:00
QHash < QString , QString > idmap ; // Temporary device ID properties hash
2019-12-24 02:44:10 +00:00
2020-05-16 01:27:08 +00:00
QString importPath ( dirpath ) ;
importPath = importPath . replace ( " \\ " , " / " ) ;
2019-12-24 02:44:10 +00:00
// Strip off end "/" if any
2020-05-16 01:27:08 +00:00
if ( importPath . endsWith ( " / " ) ) {
importPath = importPath . section ( " / " , 0 , - 2 ) ;
2019-12-24 02:44:10 +00:00
}
2020-05-16 01:27:08 +00:00
// Strip off DATALOG from importPath, and set newimportPath to the importPath containing DATALOG
if ( importPath . endsWith ( RMS9_STR_datalog ) ) {
datalogPath = importPath + " / " ;
importPath = importPath . section ( " / " , 0 , - 2 ) ;
2019-12-24 02:44:10 +00:00
} else {
2020-05-16 01:27:08 +00:00
datalogPath = importPath + " / " + RMS9_STR_datalog + " / " ;
2019-12-24 02:44:10 +00:00
}
// Add separator back
2020-05-16 01:27:08 +00:00
importPath + = " / " ;
2019-12-24 02:44:10 +00:00
// Check DATALOG folder exists and is readable
if ( ! QDir ( ) . exists ( datalogPath ) ) {
2020-05-04 23:59:59 +00:00
qDebug ( ) < < " Missing DATALOG in " < < dirpath ;
2014-09-01 04:49:05 +00:00
return - 1 ;
2019-12-24 02:44:10 +00:00
}
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
m_abort = false ;
MachineInfo info = newInfo ( ) ;
2021-06-07 02:57:52 +00:00
if ( ! parseIdentFile ( importPath , & info , idmap ) ) {
qDebug ( ) < < " Failed to parse Identification file " ;
2019-12-24 02:44:10 +00:00
return - 1 ;
2020-05-04 23:59:59 +00:00
}
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Info: " < < info . series < < info . model < < info . modelnumber < < info . serial ;
2020-04-11 15:26:03 +00:00
# ifdef IDENT_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " IdMap size: " < < idmap . size ( ) ;
foreach ( QString st , idmap . keys ( ) ) {
qDebug ( ) < < " Key " < < st < < " Value " < < idmap [ st ] ;
}
2019-12-30 16:14:05 +00:00
# endif
2019-12-24 02:44:10 +00:00
// Abort if no serial number
if ( info . serial . isEmpty ( ) ) {
2020-05-04 23:59:59 +00:00
qDebug ( ) < < " ResMed Data card is missing serial number in Indentification.tgt " ;
2019-12-24 02:44:10 +00:00
return - 1 ;
}
2014-09-01 04:49:05 +00:00
2020-10-05 21:52:14 +00:00
bool compress_backups = p_profile - > session - > compressBackupData ( ) ;
2022-02-27 16:50:10 +00:00
// Early check for STR.edf file, so we can early exit before creating faulty device record.
2020-10-06 16:37:29 +00:00
// str.edf is the first (primary) file to check, str.edf.gz is the secondary
QString pripath = importPath + " STR.edf " ; // STR.edf file
QString secpath = pripath + STR_ext_gz ; // STR.edf.gz file
QString strpath ;
2014-09-01 04:49:05 +00:00
2020-10-06 16:37:29 +00:00
// If compression is enabled, swap primary and secondary paths
2020-10-05 21:52:14 +00:00
if ( compress_backups ) {
2020-10-06 16:37:29 +00:00
strpath = pripath ;
pripath = secpath ;
secpath = strpath ;
2020-10-05 21:52:14 +00:00
}
2014-09-01 04:49:05 +00:00
2020-10-06 16:37:29 +00:00
// Check if primary path exists
QFile f ( pripath ) ;
if ( f . exists ( ) ) {
strpath = pripath ;
// If no primary file, check for secondary
} else {
f . setFileName ( secpath ) ;
strpath = secpath ;
2019-12-24 02:44:10 +00:00
if ( ! f . exists ( ) ) {
qDebug ( ) < < " Missing STR.edf file " ;
return - 1 ;
}
}
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
2022-02-27 16:50:10 +00:00
// Create device object (unless it's already registered)
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
2014-09-01 04:49:05 +00:00
2022-02-27 16:50:10 +00:00
QDate firstImportDay = QDate ( ) . fromString ( " 2010-01-01 " , " yyyy-MM-dd " ) ; // Before Series 8 devices (I think)
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
Machine * mach = p_profile - > lookupMachine ( info . serial , info . loadername ) ;
2022-02-27 16:50:10 +00:00
if ( mach ) { // we have seen this device
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " We have seen this machime " ;
2020-02-17 02:12:54 +00:00
mach - > setInfo ( info ) ; // update info
2022-02-27 16:50:10 +00:00
QDate lastDate = mach - > LastDay ( ) ; // use the last day for this device
2020-08-08 18:17:54 +00:00
firstImportDay = lastDate ; // re-import the last day, to pick up partial days
2020-08-23 22:30:38 +00:00
QDate purgeDate = mach - > purgeDate ( ) ;
if ( purgeDate . isValid ( ) ) {
firstImportDay = min ( firstImportDay , purgeDate ) ;
}
// firstImportDay = lastDate.addDays(-1); // start the day before, to pick up partial days
2020-08-08 18:17:54 +00:00
// firstImportDay = lastDate.addDays(1); // start the day after until we figure out the purge
2019-12-24 02:44:10 +00:00
} else { // Starting from new beginnings - new or purged
2022-02-27 16:50:10 +00:00
qDebug ( ) < < " New device or just purged " ;
2020-02-07 02:34:04 +00:00
p_profile - > forceResmedPrefs ( ) ;
2021-08-14 21:59:35 +00:00
int modelNum = info . modelnumber . toInt ( ) ;
if ( modelNum > = 39000 ) {
if ( ! AS11TestedModels . contains ( modelNum ) ) {
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " Device Untested " ) ,
QObject : : tr ( " Your ResMed CPAP device (Model %1) has not been tested yet. " ) . arg ( info . modelnumber ) + " \n \n " +
QObject : : tr ( " It seems similar enough to other devices that it might work, but the developers would like a .zip copy of this device's SD card to make sure it works with OSCAR. " )
2021-08-14 21:59:35 +00:00
, QMessageBox : : Ok ) ;
}
}
2019-12-24 02:44:10 +00:00
mach = p_profile - > CreateMachine ( info ) ;
}
2020-02-06 14:13:16 +00:00
QDateTime ignoreBefore = p_profile - > session - > ignoreOlderSessionsDate ( ) ;
bool ignoreOldSessions = p_profile - > session - > ignoreOlderSessions ( ) ;
2020-08-08 18:17:54 +00:00
if ( ignoreOldSessions & & ( ignoreBefore . date ( ) > firstImportDay ) )
2020-02-06 14:13:16 +00:00
firstImportDay = ignoreBefore . date ( ) ;
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " First day to import: " < < firstImportDay . toString ( ) ;
2020-05-17 23:24:31 +00:00
bool rebuild_from_backups = false ;
2019-12-24 02:44:10 +00:00
bool create_backups = p_profile - > session - > backupCardData ( ) ;
QString backup_path = mach - > getBackupPath ( ) ;
2020-10-06 01:29:01 +00:00
// Compare QDirs rather than QStrings because separators may be different, especially on Windows.
// We want to check whether import and backup paths are the same, regardless of variations in the string representations.
2020-10-05 21:52:14 +00:00
QDir ipath ( importPath ) ;
QDir bpath ( backup_path ) ;
if ( ipath = = bpath ) {
2019-12-24 02:44:10 +00:00
// Don't create backups if importing from backup folder
2020-05-17 23:24:31 +00:00
rebuild_from_backups = true ;
2019-12-24 02:44:10 +00:00
create_backups = false ;
}
///////////////////////////////////////////////////////////////////////////////////
2022-02-27 16:50:10 +00:00
// Copy the idmap into device objects properties, (overwriting any old values)
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
for ( auto i = idmap . begin ( ) , idend = idmap . end ( ) ; i ! = idend ; i + + ) {
2021-10-26 18:59:41 +00:00
mach - > info . properties [ i . key ( ) ] = i . value ( ) ;
2019-12-24 02:44:10 +00:00
}
2020-05-16 01:27:08 +00:00
///////////////////////////////////////////////////////////////////////////////////
2020-05-17 23:24:31 +00:00
// Create the backup folder structure for storing a copy of everything in..
2020-05-16 01:27:08 +00:00
// (Unless we are importing from this backup folder)
///////////////////////////////////////////////////////////////////////////////////
QDir dir ;
if ( create_backups ) {
if ( ! dir . exists ( backup_path ) ) {
2020-05-17 23:24:31 +00:00
if ( ! dir . mkpath ( backup_path ) ) {
2020-08-13 23:32:34 +00:00
qWarning ( ) < < " Could not create ResMed backup directory " < < backup_path ;
2020-05-16 01:27:08 +00:00
}
}
2020-05-17 23:24:31 +00:00
// Create the STR_Backup folder if it doesn't exist
QString strBackupPath = backup_path + " STR_Backup " ;
if ( ! dir . exists ( strBackupPath ) )
2020-08-13 23:32:34 +00:00
if ( ! dir . mkpath ( strBackupPath ) )
qWarning ( ) < < " Could not create ResMed STR backup directory " < < strBackupPath ;
2020-05-17 23:24:31 +00:00
QString newpath = backup_path + " DATALOG " ;
if ( ! dir . exists ( newpath ) )
2020-08-13 23:32:34 +00:00
if ( ! dir . mkpath ( newpath ) )
qWarning ( ) < < " Could not create ResMed DATALOG backup directory " < < newpath ;
2020-05-16 01:27:08 +00:00
2020-08-13 23:32:34 +00:00
// Copy Identification files to backup folder
2021-06-07 17:58:04 +00:00
QString idfile_ext ;
if ( QFile ( importPath + RMS9_STR_idfile + STR_ext_TGT ) . exists ( ) ) {
idfile_ext = STR_ext_TGT ;
}
else if ( QFile ( importPath + RMS9_STR_idfile + STR_ext_JSON ) . exists ( ) ) {
idfile_ext = STR_ext_JSON ;
}
else {
idfile_ext = " " ; // should never happen...
}
QFile backupFile ( backup_path + RMS9_STR_idfile + idfile_ext ) ;
2020-08-13 23:32:34 +00:00
if ( backupFile . exists ( ) )
backupFile . remove ( ) ;
2021-06-07 17:58:04 +00:00
if ( ! QFile : : copy ( importPath + RMS9_STR_idfile + idfile_ext , backup_path + RMS9_STR_idfile + idfile_ext ) )
qWarning ( ) < < " Could not copy " < < importPath + RMS9_STR_idfile + idfile_ext < < " to backup " < < backupFile . fileName ( ) ;
2020-08-13 23:32:34 +00:00
backupFile . setFileName ( backup_path + RMS9_STR_idfile + STR_ext_CRC ) ;
if ( backupFile . exists ( ) )
backupFile . remove ( ) ;
2021-06-07 17:58:04 +00:00
if ( ! QFile : : copy ( importPath + RMS9_STR_idfile + STR_ext_CRC , backup_path + RMS9_STR_idfile + STR_ext_CRC ) )
2020-08-13 23:32:34 +00:00
qWarning ( ) < < " Could not copy " < < importPath + RMS9_STR_idfile + STR_ext_CRC < < " to backup " < < backup_path ;
2020-05-16 01:27:08 +00:00
}
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
2020-05-17 23:24:31 +00:00
// Open and Process STR.edf files (including those listed in STR_Backup)
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
resdayList . clear ( ) ;
emit updateMessage ( QObject : : tr ( " Locating STR.edf File(s) . . . " )) ;
QCoreApplication : : processEvents ( ) ;
// List all STR.edf backups and tag on latest for processing
QMap < QDate , STRFile > STRmap ;
2020-05-17 23:24:31 +00:00
if ( ( ! rebuild_from_backups ) /* && create_backups */ ) {
2020-05-18 21:42:37 +00:00
// first we copy any STR_yyyymmdd.edf files and the Backup/STR.edf into STR_Backup and the STRmap
2020-05-17 23:24:31 +00:00
backupSTRfiles ( strpath , importPath , backup_path , info , STRmap ) ;
2020-05-18 21:42:37 +00:00
//Then we copy the new imported STR.edf into Backup/STR.edf and add it to the STRmap
2020-05-17 23:24:31 +00:00
QString importFile ( importPath + " STR.edf " ) ;
QString backupFile ( backup_path + " STR.edf " ) ;
2020-05-18 21:42:37 +00:00
ResMedEDFInfo * stredf = fetchSTRandVerify ( importFile , info . serial ) ;
if ( stredf ! = nullptr ) {
bool addToSTRmap = true ;
QDate date = stredf - > edfHdr . startdate_orig . date ( ) ;
long int days = stredf - > GetNumDataRecords ( ) ;
2020-08-08 18:17:54 +00:00
qDebug ( ) < < importFile . section ( " / " , - 3 , - 1 ) < < " starts at " < < date < < " for " < < days < < " ends " < < date . addDays ( days - 1 ) ;
2020-08-03 18:01:04 +00:00
if ( STRmap . contains ( date ) ) { // Keep the longer of the two STR files - or newer if equal!
2020-08-20 00:12:41 +00:00
qDebug ( ) . noquote ( ) < < importFile . section ( " / " , - 3 , - 1 ) < < " overlaps " < < STRmap [ date ] . filename . section ( " / " , - 3 , - 1 ) < < " for " < < days < < " days, ends " < < date . addDays ( days - 1 ) ;
2020-08-03 18:01:04 +00:00
if ( days > = STRmap [ date ] . days ) {
2020-08-13 23:32:34 +00:00
qDebug ( ) < < " Removing " < < STRmap [ date ] . filename . section ( " / " , - 3 , - 1 ) < < " with " < < STRmap [ date ] . days < < " days from STRmap " ;
2020-05-18 21:42:37 +00:00
STRmap . remove ( date ) ;
} else {
qDebug ( ) < < " Skipping " < < importFile . section ( " / " , - 3 , - 1 ) ;
2020-08-13 23:32:34 +00:00
qWarning ( ) < < " New import str.edf file is shorter than exisiting files - should never happen " ;
2020-05-18 21:42:37 +00:00
delete stredf ;
addToSTRmap = false ;
}
}
if ( addToSTRmap ) {
if ( compress_backups ) {
backupFile + = " .gz " ;
if ( QFile : : exists ( backupFile ) )
QFile : : remove ( backupFile ) ;
compressFile ( importFile , backupFile ) ;
}
else {
if ( QFile : : exists ( backupFile ) )
QFile : : remove ( backupFile ) ;
if ( ! QFile : : copy ( importFile , backupFile ) )
qWarning ( ) < < " Failed to copy " < < importFile < < " to " < < backupFile ;
}
STRmap [ date ] = STRFile ( backupFile , days , stredf ) ;
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Adding " < < importFile < < " to STRmap as " < < backupFile ;
2020-08-13 23:32:34 +00:00
2020-05-18 21:42:37 +00:00
// Meh.. these can be calculated if ever needed for ResScan SDcard export
2020-08-13 23:32:34 +00:00
QFile sourcePath ( importPath + " STR.crc " ) ;
if ( sourcePath . exists ( ) ) {
QFile backupFile ( backup_path + " STR.crc " ) ;
if ( backupFile . exists ( ) )
if ( ! backupFile . remove ( ) )
2020-09-01 05:02:32 +00:00
qWarning ( ) < < " Failed to remove " < < backupFile . fileName ( ) ;
2020-08-13 23:32:34 +00:00
if ( ! QFile : : copy ( importPath + " STR.crc " , backup_path + " STR.crc " ) )
qWarning ( ) < < " Failed to copy STR.crc from " < < importPath < < " to " < < backup_path ;
}
2020-05-18 21:42:37 +00:00
}
2020-05-17 23:24:31 +00:00
}
2020-05-16 01:27:08 +00:00
} else { // get the STR file that is in the BACKUP folder that we are rebuilding from
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Rebuilding from BACKUP folder " ;
2020-05-17 23:24:31 +00:00
ResMedEDFInfo * stredf = fetchSTRandVerify ( strpath , info . serial ) ;
if ( stredf ! = nullptr ) {
QDate date = stredf - > edfHdr . startdate_orig . date ( ) ;
long int days = stredf - > GetNumDataRecords ( ) ;
qDebug ( ) < < strpath . section ( " / " , - 2 , - 1 ) < < " starts at " < < date < < " for " < < days < < " ends " < < date . addDays ( days - 1 ) ;
STRmap [ date ] = STRFile ( strpath , days , stredf ) ;
2020-04-27 00:59:23 +00:00
} else {
qDebug ( ) < < " Failed to open " < < strpath ;
}
} // end if not importing the backup files
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " STRmap size is " < < STRmap . size ( ) ;
2020-02-08 01:16:47 +00:00
# endif
2019-12-24 02:44:10 +00:00
2020-05-16 01:27:08 +00:00
// Now we open the REAL destination STR_Backup, and open the rest for later parsing
2019-12-24 02:44:10 +00:00
dir . setPath ( backup_path + " STR_Backup " ) ;
dir . setFilter ( QDir : : Files | QDir : : Hidden | QDir : : Readable ) ;
QFileInfoList flist = dir . entryInfoList ( ) ;
QDate date ;
2020-04-27 00:59:23 +00:00
long int days ;
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " STR_Backup folder size is " < < flist . size ( ) ;
2020-02-08 01:16:47 +00:00
# endif
2019-12-24 02:44:10 +00:00
2020-05-18 21:42:37 +00:00
qDebug ( ) < < " Add files in STR_Backup to STRmap (unless they are already there) " ;
2019-12-24 02:44:10 +00:00
// Add any STR_Backup versions to the file list
for ( auto & fi : flist ) {
QString filename = fi . fileName ( ) ;
2020-08-08 18:17:54 +00:00
if ( ! filename . startsWith ( " STR " , Qt : : CaseInsensitive ) )
2019-12-24 02:44:10 +00:00
continue ;
2020-08-08 18:17:54 +00:00
if ( ! ( filename . endsWith ( " edf.gz " , Qt : : CaseInsensitive ) | | filename . endsWith ( " edf " , Qt : : CaseInsensitive ) ) )
2019-12-24 02:44:10 +00:00
continue ;
2020-04-29 17:57:10 +00:00
QString datestr = filename . section ( " STR- " , - 1 ) . section ( " .edf " , 0 , 0 ) ; // +"01";
2014-09-01 04:49:05 +00:00
2020-05-17 23:24:31 +00:00
ResMedEDFInfo * stredf = fetchSTRandVerify ( fi . canonicalFilePath ( ) , info . serial ) ;
if ( stredf = = nullptr )
2019-12-24 02:44:10 +00:00
continue ;
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
// Don't trust the filename date, pick the one inside the STR...
date = stredf - > edfHdr . startdate_orig . date ( ) ;
2020-04-27 00:59:23 +00:00
days = stredf - > GetNumDataRecords ( ) ;
if ( STRmap . contains ( date ) ) { // Keep the longer of the two STR files
2020-08-20 00:12:41 +00:00
qDebug ( ) . noquote ( ) < < fi . canonicalFilePath ( ) . section ( " / " , - 3 , - 1 ) < < " overlaps " < < STRmap [ date ] . filename . section ( " / " , - 3 , - 1 ) < < " for " < < days < < " ends " < < date . addDays ( days - 1 ) ;
2020-04-27 00:59:23 +00:00
if ( days < = STRmap [ date ] . days ) {
2020-05-18 21:42:37 +00:00
qDebug ( ) < < " Skipping " < < fi . canonicalFilePath ( ) . section ( " / " , - 3 , - 1 ) ;
2020-04-29 23:51:32 +00:00
delete stredf ;
2020-04-27 00:59:23 +00:00
continue ;
2020-05-17 23:24:31 +00:00
} else {
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Removing " < < STRmap [ date ] . filename . section ( " / " , - 3 , - 1 ) < < " from STRmap " ;
2020-05-17 23:24:31 +00:00
STRmap . remove ( date ) ;
2020-04-27 00:59:23 +00:00
}
}
2014-09-01 04:49:05 +00:00
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Adding " < < fi . canonicalFilePath ( ) . section ( " / " , - 3 , - 1 ) < < " starts at " < < date < < " for " < < days < < " to STRmap " ;
2020-04-27 00:59:23 +00:00
STRmap [ date ] = STRFile ( fi . canonicalFilePath ( ) , days , stredf ) ;
2019-12-24 02:44:10 +00:00
} // end for walking the STR_Backup directory
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Finished STRmap size is now " < < STRmap . size ( ) ;
2020-02-08 01:16:47 +00:00
# endif
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
// Build a Date map of all records in STR.edf files, populating ResDayList
///////////////////////////////////////////////////////////////////////////////////
2014-09-01 04:49:05 +00:00
2020-02-14 20:18:25 +00:00
if ( ! ProcessSTRfiles ( mach , STRmap , firstImportDay ) ) {
qCritical ( ) < < " ProcessSTR failed, abandoning this import " ;
2020-02-17 02:41:56 +00:00
return - 1 ;
2020-02-14 20:18:25 +00:00
}
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
// We are done with the Parsed STR EDF objects, so delete them
for ( auto it = STRmap . begin ( ) , end = STRmap . end ( ) ; it ! = end ; + + it ) {
2020-04-29 17:57:10 +00:00
QString fullname = it . value ( ) . filename ;
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-05-16 01:27:08 +00:00
qDebug ( ) < < " Deleting edf object of " < < fullname ;
2020-02-08 01:16:47 +00:00
# endif
2020-04-29 17:57:10 +00:00
QString datepart = fullname . section ( " STR- " , - 1 ) . section ( " .edf " , 0 , 0 ) ;
if ( datepart . size ( ) = = 6 ) { // old style name, change to full date
QFile str ( fullname ) ;
QString newdate = it . key ( ) . toString ( " yyyyMMdd " ) ;
QString newName = fullname . replace ( datepart , newdate ) ;
qDebug ( ) < < " Renaming " < < it . value ( ) . filename < < " to " < < newName ;
2020-08-13 23:32:34 +00:00
if ( ! str . rename ( newName ) )
qWarning ( ) < < " Rename Failed " ;
2020-04-29 17:57:10 +00:00
}
2019-12-24 02:44:10 +00:00
delete it . value ( ) . edf ;
}
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Finished STRmap cleanup " ;
2020-02-08 01:16:47 +00:00
# endif
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////////
// Scan DATALOG files, sort, and import any new sessions
///////////////////////////////////////////////////////////////////////////////////
2014-09-01 04:49:05 +00:00
2019-12-24 02:44:10 +00:00
// First remove a legacy file if present...
QFile impfile ( mach - > getDataPath ( ) + " /imported_files.csv " ) ;
if ( impfile . exists ( ) )
impfile . remove ( ) ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
emit updateMessage ( QObject : : tr ( " Cataloguing EDF Files... " ) ) ;
QApplication : : processEvents ( ) ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
if ( isAborted ( ) )
return 0 ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Starting scan of DATALOG " ;
2020-02-06 14:13:16 +00:00
// sleep(1);
2020-05-16 01:27:08 +00:00
dir . setPath ( datalogPath ) ;
2019-12-24 02:44:10 +00:00
ScanFiles ( mach , datalogPath , firstImportDay ) ;
if ( isAborted ( ) )
return 0 ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Finished DATALOG scan " ;
2020-02-06 14:13:16 +00:00
// sleep(1);
2019-12-24 02:44:10 +00:00
// Now at this point we have resdayList populated with processable summary and EDF files data
// that can be processed in threads..
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
emit updateMessage ( QObject : : tr ( " Queueing Import Tasks... " ) ) ;
QApplication : : processEvents ( ) ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
for ( auto rdi = resdayList . begin ( ) , rend = resdayList . end ( ) ; rdi ! = rend ; rdi + + ) {
if ( isAborted ( ) )
return 0 ;
2014-08-11 06:13:25 +00:00
2019-12-24 02:44:10 +00:00
QDate date = rdi . key ( ) ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
ResMedDay & resday = rdi . value ( ) ;
resday . date = date ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
checkSummaryDay ( resday , date , mach ) ;
}
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
sessionCount = 0 ;
emit updateMessage ( QObject : : tr ( " Importing Sessions... " ) ) ;
2020-04-29 17:57:10 +00:00
// Walk down the resDay list
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " About to call runTasks() " ;
2019-12-24 02:44:10 +00:00
runTasks ( ) ;
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Finshed runTasks() with " < < sessionCount < < " new sessions " ;
2019-12-24 02:44:10 +00:00
int num_new_sessions = sessionCount ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
////////////////////////////////////////////////////////////////////////////////////
// Now look for any new summary data that can be extracted from STR.edf records
////////////////////////////////////////////////////////////////////////////////////
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
emit updateMessage ( QObject : : tr ( " Finishing Up... " ) ) ;
QApplication : : processEvents ( ) ;
2014-07-27 16:35:49 +00:00
2020-08-13 23:32:34 +00:00
qDebug ( ) < < " About to call finishAddingSessions() " ;
2019-12-24 02:44:10 +00:00
finishAddingSessions ( ) ;
2020-08-13 23:32:34 +00:00
qDebug ( ) < < " Finshed finishedAddingSessions() with " < < sessionCount < < " new sessions " ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
# ifdef DEBUG_EFFICIENCY
{
qint64 totalbytes = 0 ;
qint64 totalns = 0 ;
qDebug ( ) < < " Performance / Efficiency Information " ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
for ( auto it = channel_efficiency . begin ( ) , end = channel_efficiency . end ( ) ; it ! = end ; it + + ) {
ChannelID code = it . key ( ) ;
qint64 value = it . value ( ) ;
qint64 ns = channel_time [ code ] ;
totalbytes + = value ;
totalns + = ns ;
double secs = double ( ns ) / 1000000000.0 L ;
QString s = value < 0 ? " saved " : " cost " ;
qDebug ( ) < < " Time-Delta conversion for " + schema : : channel [ code ] . label ( ) + " " + s + " " +
QString : : number ( qAbs ( value ) ) + " bytes and took " + QString : : number ( secs , ' f ' , 4 ) + " s " ;
}
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Total toTimeDelta function usage: " < < totalbytes < < " in " < < double ( totalns ) / 1000000000.0 < < " seconds " ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Total CPU time in EDF Open " < < timeInEDFOpen ;
qDebug ( ) < < " Total CPU time in EDF Parser " < < timeInEDFInfo ;
qDebug ( ) < < " Total CPU time in LoadBRP " < < timeInLoadBRP ;
qDebug ( ) < < " Total CPU time in LoadPLD " < < timeInLoadPLD ;
qDebug ( ) < < " Total CPU time in LoadSAD " < < timeInLoadSAD ;
qDebug ( ) < < " Total CPU time in LoadEVE " < < timeInLoadEVE ;
qDebug ( ) < < " Total CPU time in LoadCSL " < < timeInLoadCSL ;
qDebug ( ) < < " Total CPU time in (BRP) AddWaveform " < < timeInAddWaveform ;
qDebug ( ) < < " Total CPU time in TimeDelta function " < < timeInTimeDelta ;
2022-01-05 20:35:35 +00:00
channel_efficiency . clear ( ) ;
channel_time . clear ( ) ;
2019-12-24 02:44:10 +00:00
}
# endif
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
// sessfiles.clear();
// strsess.clear();
// strdate.clear();
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Total Events " < < event_cnt ;
qDebug ( ) < < " Total new Sessions " < < num_new_sessions ;
2014-08-11 06:13:25 +00:00
2020-08-23 22:30:38 +00:00
mach - > clearPurgeDate ( ) ;
2019-12-24 02:44:10 +00:00
return num_new_sessions ;
2020-02-02 19:29:23 +00:00
} // end Open()
2014-07-27 16:35:49 +00:00
2020-05-18 17:59:10 +00:00
ResMedEDFInfo * fetchSTRandVerify ( QString filename , QString serialNumber )
2020-05-17 23:24:31 +00:00
{
ResMedEDFInfo * stredf = new ResMedEDFInfo ( ) ;
if ( ! stredf - > Open ( filename ) ) {
2020-08-13 23:32:34 +00:00
qWarning ( ) < < " Failed to open " < < filename ;
2020-05-17 23:24:31 +00:00
delete stredf ;
return nullptr ;
}
if ( ! stredf - > Parse ( ) ) {
qDebug ( ) < < " Faulty STR file " < < filename ;
delete stredf ;
return nullptr ;
}
if ( stredf - > serialnumber ! = serialNumber ) {
qDebug ( ) < < " Identification.tgt Serial number doesn't match " < < filename ;
delete stredf ;
return nullptr ;
}
return stredf ;
}
2019-12-24 02:44:10 +00:00
void StoreSettings ( Session * sess , STRRecord & R ) ; // forward
void ResmedLoader : : checkSummaryDay ( ResMedDay & resday , QDate date , Machine * mach )
{
Day * day = p_profile - > FindDay ( date , MT_CPAP ) ;
bool reimporting = false ;
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Starting checkSummary for " < < date . toString ( ) ;
2020-08-13 23:32:34 +00:00
# endif
2019-12-24 02:44:10 +00:00
if ( day & & day - > hasMachine ( mach ) ) {
2022-02-27 16:50:10 +00:00
// Sessions found for this device, check if only summary info
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Sessions already found for this date " ;
2020-08-13 23:32:34 +00:00
# endif
2019-12-24 02:44:10 +00:00
if ( day - > summaryOnly ( mach ) & & ( resday . files . size ( ) > 0 ) ) {
// Note: if this isn't an EDF file, there's really no point doing this here,
// but the worst case scenario is this session is deleted and reimported.. this just slows things down a bit in that case
// This day was first imported as a summary from STR.edf, so we now totally want to redo this day
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Summary sessions only - delete them " ;
2020-08-13 23:32:34 +00:00
# endif
2019-12-24 02:44:10 +00:00
QList < Session * > sessions = day - > getSessions ( MT_CPAP ) ;
for ( auto & sess : sessions ) {
day - > removeSession ( sess ) ;
delete sess ;
}
} else if ( day - > noSettings ( mach ) & & resday . str . date . isValid ( ) ) {
// STR is present now, it wasn't before... we don't need to trash the files, but we do want the official settings.
// Do it right here
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Date was missing settings, now we have them " ;
2020-08-13 23:32:34 +00:00
# endif
2019-12-24 02:44:10 +00:00
for ( auto & sess : day - > sessions ) {
if ( sess - > machine ( ) ! = mach )
continue ;
2020-04-11 15:26:03 +00:00
# ifdef STR_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Adding STR.edf information to session " < < sess - > session ( ) ;
2020-04-11 15:26:03 +00:00
# endif
2019-12-24 02:44:10 +00:00
StoreSettings ( sess , resday . str ) ;
sess - > setNoSettings ( false ) ;
sess - > SetChanged ( true ) ;
sess - > StoreSummary ( ) ;
}
2020-08-08 18:17:54 +00:00
} else {
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Have summary and details for this date! " ;
2020-08-13 23:32:34 +00:00
# endif
int numPairs = 0 ;
for ( int i = 0 ; i < resday . str . maskevents / 2 ; i + + )
if ( resday . str . maskon [ i ] ! = resday . str . maskoff [ i ] )
numPairs + + ;
2020-08-13 00:26:14 +00:00
QList < Session * > sessions = day - > getSessions ( MT_CPAP , true ) ;
2020-08-19 17:14:02 +00:00
// If we have more sessions that we found in the str file,
2022-02-27 16:50:10 +00:00
// or if the sessions are for a different device,
2020-08-19 17:14:02 +00:00
// leave well enough alone and don't re-import the day
if ( sessions . length ( ) > = numPairs | | sessions [ 0 ] - > machine ( ) ! = mach ) {
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-13 00:26:14 +00:00
qDebug ( ) < < " No new sessions -- skipping. Sessions now in day: " ;
qDebug ( ) < < " i sessionID s_first from - to " ;
for ( int i = 0 ; i < sessions . length ( ) ; i + + ) {
qDebug ( ) . noquote ( ) < < i < < sessions [ i ] - > session ( )
< < sessions [ i ] - > first ( )
< < QDateTime : : fromMSecsSinceEpoch ( sessions [ i ] - > first ( ) ) . toString ( " hh:mm:ss " )
< < " - " < < QDateTime : : fromMSecsSinceEpoch ( sessions [ i ] - > last ( ) ) . toString ( " hh:mm:ss " ) ;
}
2020-08-13 23:32:34 +00:00
# endif
2020-08-08 18:17:54 +00:00
return ;
}
2020-08-19 17:14:02 +00:00
qDebug ( ) < < " Maskevent count/2 (modified) " < < numPairs < < " is greater than the existing MT_CPAP session count " < < sessions . length ( ) ;
qDebug ( ) . noquote ( ) < < " Purging and re-importing " < < day - > date ( ) . toString ( ) ;
2020-08-08 18:17:54 +00:00
for ( auto & sess : sessions ) {
day - > removeSession ( sess ) ;
delete sess ;
}
}
2019-12-24 02:44:10 +00:00
}
2014-08-20 03:03:01 +00:00
2020-02-10 21:04:03 +00:00
ResDayTask * rdt = new ResDayTask ( this , mach , & resday , saveCallback ) ;
2019-12-24 02:44:10 +00:00
rdt - > reimporting = reimporting ;
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " in checkSummary, Queue task for " < < resday . date . toString ( ) ;
2020-08-13 23:32:34 +00:00
# endif
2019-12-24 02:44:10 +00:00
queTask ( rdt ) ;
2014-07-27 16:35:49 +00:00
}
2018-04-28 05:33:26 +00:00
///////////////////////////////////////////////////////////////////////////////////////////
// Sorted EDF files that need processing into date records according to ResMed noon split
///////////////////////////////////////////////////////////////////////////////////////////
2019-12-24 02:44:10 +00:00
int ResmedLoader : : ScanFiles ( Machine * mach , const QString & datalog_path , QDate firstImport )
2014-07-27 16:35:49 +00:00
{
2022-01-10 23:02:25 +00:00
# ifdef DEBUG_EFFICIENCY
2018-04-28 05:33:26 +00:00
QTime time ;
2022-01-10 23:02:25 +00:00
# endif
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
bool create_backups = p_profile - > session - > backupCardData ( ) ;
2014-07-28 13:56:29 +00:00
QString backup_path = mach - > getBackupPath ( ) ;
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
if ( datalog_path = = ( backup_path + RMS9_STR_datalog + " / " ) ) {
2014-07-27 16:35:49 +00:00
// Don't create backups if importing from backup folder
create_backups = false ;
}
2018-04-28 05:33:26 +00:00
///////////////////////////////////////////////////////////////////////////////////////
// Generate list of files for later processing
///////////////////////////////////////////////////////////////////////////////////////
2019-08-03 14:59:08 +00:00
qDebug ( ) < < " Generating list of EDF files " ;
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " First Import date is " < < firstImport ;
2018-04-25 18:04:05 +00:00
2018-04-28 05:33:26 +00:00
# ifdef DEBUG_EFFICIENCY
time . start ( ) ;
# endif
2014-07-27 16:35:49 +00:00
QDir dir ( datalog_path ) ;
2018-05-04 10:22:55 +00:00
2022-02-27 16:50:10 +00:00
// First list any EDF files in DATALOG folder - Series 9 devices
2018-05-04 10:22:55 +00:00
QStringList filter ;
filter < < " *.edf " ;
dir . setNameFilters ( filter ) ;
QFileInfoList EDFfiles = dir . entryInfoList ( ) ;
2019-08-16 23:50:09 +00:00
// Scan through all folders looking for EDF files, skip any already imported and peek inside to get durations
2018-05-04 10:22:55 +00:00
dir . setNameFilters ( QStringList ( ) ) ;
2014-07-27 16:35:49 +00:00
dir . setFilter ( QDir : : Dirs | QDir : : Hidden | QDir : : NoDotAndDotDot ) ;
QString filename ;
2018-04-28 05:33:26 +00:00
bool ok ;
2014-07-27 16:35:49 +00:00
2019-08-03 14:59:08 +00:00
QFileInfoList dirlist = dir . entryInfoList ( ) ;
int dirlistSize = dirlist . size ( ) ;
2014-07-27 16:35:49 +00:00
2019-08-30 17:22:19 +00:00
// QDateTime ignoreBefore = p_profile->session->ignoreOlderSessionsDate();
// bool ignoreOldSessions = p_profile->session->ignoreOlderSessions();
2019-08-03 14:59:08 +00:00
2018-04-28 05:33:26 +00:00
// Scan for any sub folders and create files lists
2019-08-03 14:59:08 +00:00
for ( int i = 0 ; i < dirlistSize ; i + + ) {
const QFileInfo & fi = dirlist . at ( i ) ;
2014-07-27 16:35:49 +00:00
filename = fi . fileName ( ) ;
2018-04-28 05:33:26 +00:00
int len = filename . length ( ) ;
2020-04-29 17:57:10 +00:00
if ( len = = 4 ) { // This is a year folder in BackupDATALOG
2014-07-27 16:35:49 +00:00
filename . toInt ( & ok ) ;
2019-08-03 14:59:08 +00:00
if ( ! ok ) {
2020-02-02 19:29:23 +00:00
qDebug ( ) < < " Skipping directory - bad 4-letter name " < < filename ;
continue ;
}
} else if ( len = = 8 ) { // test directory date
QDate dirDate = QDate ( ) . fromString ( filename , " yyyyMMdd " ) ;
if ( dirDate < firstImport ) {
# ifdef SESSION_DEBUG
qDebug ( ) < < " Skipping directory - ignore before " < < filename ;
# endif
continue ;
2019-08-03 14:59:08 +00:00
}
} else {
2020-02-02 19:29:23 +00:00
qDebug ( ) < < " Skipping directory - bad name size " < < filename ;
continue ;
2014-07-27 16:35:49 +00:00
}
2019-08-03 14:59:08 +00:00
// Get file lists under this directory
dir . setPath ( fi . canonicalFilePath ( ) ) ;
dir . setFilter ( QDir : : Files | QDir : : Hidden | QDir : : NoSymLinks ) ;
dir . setSorting ( QDir : : Name ) ;
// Append all files to one big QFileInfoList
EDFfiles . append ( dir . entryInfoList ( ) ) ;
2014-07-27 16:35:49 +00:00
}
2018-04-28 05:33:26 +00:00
# ifdef DEBUG_EFFICIENCY
qDebug ( ) < < " Generating EDF files list took " < < time . elapsed ( ) < < " ms " ;
# endif
2019-08-16 23:50:09 +00:00
qDebug ( ) < < " EDFfiles list size is " < < EDFfiles . size ( ) ;
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
////////////////////////////////////////////////////////////////////////////////////////
// Scan through EDF files, Extracting EDF Durations, and skipping already imported files
// Check for duplicates along the way from compressed/uncompressed files
////////////////////////////////////////////////////////////////////////////////////////
2015-08-02 05:53:09 +00:00
2018-04-28 05:33:26 +00:00
# ifdef DEBUG_EFFICIENCY
2018-04-22 20:04:39 +00:00
time . start ( ) ;
2018-04-28 05:33:26 +00:00
# endif
QString datestr ;
QDateTime datetime ;
QDate date ;
int totalfiles = EDFfiles . size ( ) ;
2019-07-22 21:01:47 +00:00
qDebug ( ) < < " Scanning " < < totalfiles < < " EDF files " ;
2018-04-22 20:04:39 +00:00
// Calculate number of files for progress bar for this stage
2018-04-28 05:33:26 +00:00
int pbarFreq = totalfiles / 50 ;
2019-07-22 21:01:47 +00:00
if ( pbarFreq < 1 ) // stop a divide by zero
pbarFreq = 1 ;
2018-04-22 20:04:39 +00:00
2018-05-07 18:42:23 +00:00
emit setProgressValue ( 0 ) ;
emit setProgressMax ( totalfiles ) ;
QCoreApplication : : processEvents ( ) ;
2018-04-22 20:04:39 +00:00
2018-04-28 05:33:26 +00:00
qDebug ( ) < < " Starting EDF duration scan pass " ;
for ( int i = 0 ; i < totalfiles ; + + i ) {
2019-07-22 21:01:47 +00:00
if ( isAborted ( ) )
return 0 ;
2018-05-07 01:57:58 +00:00
2018-04-28 05:33:26 +00:00
const QFileInfo & fi = EDFfiles . at ( i ) ;
2018-04-22 20:04:39 +00:00
2018-04-28 05:33:26 +00:00
// Update progress bar
2018-05-07 18:42:23 +00:00
if ( ( i % pbarFreq ) = = 0 ) {
emit setProgressValue ( i ) ;
QCoreApplication : : processEvents ( ) ;
2018-04-28 05:33:26 +00:00
}
2018-04-22 20:04:39 +00:00
2018-04-28 05:33:26 +00:00
// Forget about it if it can't be read.
if ( ! fi . isReadable ( ) ) {
2019-09-16 18:53:27 +00:00
qWarning ( ) < < fi . fileName ( ) < < " is unreadable and has been ignored " ;
continue ;
}
// Skip empty files
if ( fi . size ( ) = = 0 ) {
qWarning ( ) < < fi . fileName ( ) < < " is empty and has been ignored " ;
2018-04-28 05:33:26 +00:00
continue ;
2019-09-22 21:49:00 +00:00
}
2014-08-26 07:55:01 +00:00
2018-04-28 05:33:26 +00:00
filename = fi . fileName ( ) ;
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
datestr = filename . section ( " _ " , 0 , 1 ) ;
datetime = QDateTime ( ) . fromString ( datestr , " yyyyMMdd_HHmmss " ) ;
date = datetime . date ( ) ;
// ResMed splits days at noon and now so do we, so all times before noon
// go to the previous day
if ( datetime . time ( ) . hour ( ) < 12 ) {
date = date . addDays ( - 1 ) ;
}
2014-07-27 16:35:49 +00:00
2020-08-08 18:17:54 +00:00
if ( date < firstImport ) {
# ifdef SESSION_DEBUG
qDebug ( ) < < " Skipping file - ignore before " < < filename ;
# endif
2020-02-02 19:29:23 +00:00
continue ;
2020-08-08 18:17:54 +00:00
}
2018-05-07 01:30:42 +00:00
2018-04-28 05:33:26 +00:00
// Chop off the .gz component if it exists, it's not needed at this stage
if ( filename . endsWith ( STR_ext_gz ) ) {
filename . chop ( 3 ) ;
}
2018-05-03 09:59:31 +00:00
QString fullpath = fi . filePath ( ) ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
QString newpath = create_backups ? Backup ( fullpath , backup_path ) : fullpath ;
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
// Accept only .edf and .edf.gz files
2019-07-22 21:01:47 +00:00
if ( filename . right ( 4 ) . toLower ( ) ! = ( " . " + STR_ext_EDF ) )
2018-04-28 05:33:26 +00:00
continue ;
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
// QString ext = key.section("_", -1).section(".",0,0).toUpper();
// EDFType type = lookupEDFType(ext);
2014-10-08 16:51:09 +00:00
2018-04-28 05:33:26 +00:00
// Find or create ResMedDay object for this date
2018-05-05 21:58:11 +00:00
auto rd = resdayList . find ( date ) ;
2018-04-28 05:33:26 +00:00
if ( rd = = resdayList . end ( ) ) {
2019-09-25 11:25:44 +00:00
rd = resdayList . insert ( date , ResMedDay ( date ) ) ;
2018-04-28 05:33:26 +00:00
rd . value ( ) . date = date ;
2014-07-27 16:35:49 +00:00
2018-04-28 05:33:26 +00:00
// We have data files without STR.edf record... the user MAY be planning on importing from another backup
// later which could cause problems if we don't deal with it.
// Best solution I can think of is import and tag the day No Settings and skip the day from overview.
}
ResMedDay & resday = rd . value ( ) ;
2014-07-27 16:35:49 +00:00
2019-07-22 21:01:47 +00:00
if ( ! resday . files . contains ( filename ) ) {
2018-04-28 05:33:26 +00:00
resday . files [ filename ] = newpath ;
}
}
2018-05-03 09:59:31 +00:00
# ifdef DEBUG_EFFICIENCY
qDebug ( ) < < " Scanning EDF files took " < < time . elapsed ( ) < < " ms " ;
# endif
2019-07-22 21:01:47 +00:00
qDebug ( ) < < " resdayList size is " < < resdayList . size ( ) ;
2019-12-24 02:44:10 +00:00
return resdayList . size ( ) ;
} // end of scanFiles
2018-06-07 22:03:20 +00:00
2019-12-24 02:44:10 +00:00
QString ResmedLoader : : Backup ( const QString & fullname , const QString & backup_path )
2018-05-03 05:08:45 +00:00
{
2019-12-24 02:44:10 +00:00
QDir dir ;
QString filename , yearstr , newname , oldname ;
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
bool compress = p_profile - > session - > compressBackupData ( ) ;
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
bool ok ;
bool gz = ( fullname . right ( 3 ) . toLower ( ) = = STR_ext_gz ) ; // Input file is a .gz?
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
filename = fullname . section ( " / " , - 1 ) ;
if ( gz ) {
filename . chop ( 3 ) ;
2018-05-03 05:08:45 +00:00
}
2019-12-24 02:44:10 +00:00
yearstr = filename . left ( 4 ) ;
yearstr . toInt ( & ok , 10 ) ;
2018-05-03 05:08:45 +00:00
2020-08-09 21:17:21 +00:00
if ( ! ok ) {
2020-04-29 17:57:10 +00:00
qDebug ( ) < < " Invalid EDF filename given to ResMedLoader::Backup() " < < fullname ;
2019-12-24 02:44:10 +00:00
return " " ;
2018-05-31 01:11:40 +00:00
}
2018-04-28 05:33:26 +00:00
2020-05-04 23:59:59 +00:00
QString newpath = backup_path + " DATALOG " + " / " + yearstr ;
if ( ! dir . exists ( newpath ) )
dir . mkpath ( newpath ) ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
newname = newpath + " / " + filename ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
QString tmpname = newname ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
QString newnamegz = newname + STR_ext_gz ;
QString newnamenogz = newname ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
newname = compress ? newnamegz : newnamenogz ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
// First make sure the correct backup exists in the right place
2020-08-09 21:17:21 +00:00
// Allow for second import of newer version of EVE and CSL edf files
2020-08-19 17:14:02 +00:00
// But don't try to copy onto itself (as when rebuilding CPAP data from backup)
2020-10-06 01:29:01 +00:00
// Compare QDirs rather than QStrings to handle variations in separators, etc.
2020-10-05 21:52:14 +00:00
QFile nf ( newname ) ;
QFile of ( fullname ) ;
QFileInfo nfi ( nf ) ;
QFileInfo ofi ( of ) ;
QDir nfdir = nfi . dir ( ) ;
QDir ofdir = ofi . dir ( ) ;
if ( nfdir ! = ofdir ) {
2020-08-19 17:14:02 +00:00
if ( QFile : : exists ( newname ) ) // remove existing backup
QFile : : remove ( newname ) ;
if ( compress ) {
// If input file is already compressed.. copy it to the right location, otherwise compress it
if ( gz ) {
if ( ! QFile : : copy ( fullname , newname ) )
qWarning ( ) < < " unable to copy " < < fullname < < " to " < < newname ;
}
else
compressFile ( fullname , newname ) ;
} else {
// If inputs a gz, uncompress it, otherwise copy is raw
if ( gz )
uncompressFile ( fullname , newname ) ;
else {
if ( ! QFile : : copy ( fullname , newname ) )
qWarning ( ) < < " unable to copy " < < fullname < < " to " < < newname ;
}
2020-08-13 23:32:34 +00:00
}
2020-08-09 21:17:21 +00:00
}
2019-12-24 02:44:10 +00:00
2020-08-09 21:17:21 +00:00
// Now the correct backup is in place, we can trash any unneeded backup
2019-12-24 02:44:10 +00:00
if ( compress ) {
// Remove any uncompressed duplicate
2020-08-09 21:17:21 +00:00
if ( QFile : : exists ( newnamenogz ) )
QFile : : remove ( newnamenogz ) ;
2019-12-24 02:44:10 +00:00
} else {
2020-08-09 21:17:21 +00:00
// Delete the compressed copy
if ( QFile : : exists ( newnamegz ) )
QFile : : remove ( newnamegz ) ;
2018-04-28 05:33:26 +00:00
}
2019-12-24 02:44:10 +00:00
// Used to store it under Backup\Datalog
// Remove any traces from old backup directory structure
2020-10-05 21:52:14 +00:00
if ( nfdir ! = ofdir ) {
oldname = backup_path + RMS9_STR_datalog + " / " + filename ;
if ( QFile : : exists ( oldname ) )
QFile : : remove ( oldname ) ;
if ( QFile : : exists ( oldname + STR_ext_gz ) )
QFile : : remove ( oldname + STR_ext_gz ) ;
}
2019-12-24 02:44:10 +00:00
return newname ;
2014-07-27 16:35:49 +00:00
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
// This function parses a list of STR files and creates a date ordered map of individual records
2020-02-14 20:18:25 +00:00
bool ResmedLoader : : ProcessSTRfiles ( Machine * mach , QMap < QDate , STRFile > & STRmap , QDate firstImport )
2018-04-28 05:33:26 +00:00
{
2021-07-25 22:43:23 +00:00
bool AS_eleven = ( mach - > info . modelnumber . toInt ( ) > = 39000 ) ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
// QDateTime ignoreBefore = p_profile->session->ignoreOlderSessionsDate();
// bool ignoreOldSessions = p_profile->session->ignoreOlderSessions();
2018-05-03 05:08:45 +00:00
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Starting ProcessSTRfiles " ;
int totalRecs = 0 ; // Count the STR days
2019-12-24 02:44:10 +00:00
for ( auto it = STRmap . begin ( ) , end = STRmap . end ( ) ; it ! = end ; + + it ) {
STRFile & file = it . value ( ) ;
ResMedEDFInfo & str = * file . edf ;
2020-04-29 17:57:10 +00:00
int days = str . GetNumDataRecords ( ) ;
totalRecs + = days ;
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-05-16 21:31:18 +00:00
qDebug ( ) < < " STR file is " < < file . filename . section ( " / " , - 3 , - 1 ) ;
2020-04-29 17:57:10 +00:00
qDebug ( ) < < " First day " < < QDateTime : : fromMSecsSinceEpoch ( str . startdate , EDFInfo : : localNoDST ) . date ( ) . toString ( ) < < " for " < < days < < " days " ;
2020-02-08 01:16:47 +00:00
# endif
2019-12-24 02:44:10 +00:00
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
emit updateMessage ( QObject : : tr ( " Parsing STR.edf records... " ) ) ;
emit setProgressMax ( totalRecs ) ;
QCoreApplication : : processEvents ( ) ;
int currentRec = 0 ;
2020-08-08 18:17:54 +00:00
// Walk through all the STR files in the STRmap
2019-12-24 02:44:10 +00:00
for ( auto it = STRmap . begin ( ) , end = STRmap . end ( ) ; it ! = end ; + + it ) {
STRFile & file = it . value ( ) ;
ResMedEDFInfo & str = * file . edf ;
QDate date = str . edfHdr . startdate_orig . date ( ) ; // each STR.edf record starts at 12 noon
int size = str . GetNumDataRecords ( ) ;
QDate lastDay = date . addDays ( size - 1 ) ;
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
QString & strfile = file . filename ;
qDebug ( ) < < " Processing " < < strfile . section ( " / " , - 3 , - 1 ) < < date . toString ( ) < < " for " < < size < < " days " ;
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Last day is " < < lastDay ;
2020-02-08 01:16:47 +00:00
# endif
2019-12-24 02:44:10 +00:00
if ( lastDay < firstImport ) {
2020-08-08 18:17:54 +00:00
# ifdef STR_DEBUG
2020-05-16 21:31:18 +00:00
qDebug ( ) < < " LastDay before firstImport, skipping " < < strfile . section ( " / " , - 3 , - 1 ) ;
2020-08-08 18:17:54 +00:00
# endif
2019-12-24 02:44:10 +00:00
continue ;
}
// ResMed and their consistent naming and spacing... :/
2022-02-27 16:50:10 +00:00
EDFSignal * maskon = str . lookupLabel ( " Mask On " ) ; // Series 9 devices
2019-12-24 02:44:10 +00:00
if ( ! maskon ) {
2022-02-27 16:50:10 +00:00
maskon = str . lookupLabel ( " MaskOn " ) ; // Series 1x devices
2019-12-24 02:44:10 +00:00
}
EDFSignal * maskoff = str . lookupLabel ( " Mask Off " ) ;
if ( ! maskoff ) {
maskoff = str . lookupLabel ( " MaskOff " ) ;
2018-05-03 05:08:45 +00:00
}
2020-02-06 14:13:16 +00:00
EDFSignal * maskeventcount = str . lookupLabel ( " Mask Events " ) ;
2020-08-03 18:01:04 +00:00
if ( ! maskeventcount ) {
2020-02-06 14:13:16 +00:00
maskeventcount = str . lookupLabel ( " MaskEvents " ) ;
}
2020-02-14 20:18:25 +00:00
if ( ! maskon | | ! maskoff | | ! maskeventcount ) {
qCritical ( ) < < " Corrupt or untranslated STR.edf file " ;
return false ;
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
EDFSignal * sig = nullptr ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
// For each data record, representing 1 day each
for ( int rec = 0 ; rec < size ; + + rec , date = date . addDays ( 1 ) ) {
emit setProgressValue ( + + currentRec ) ;
QCoreApplication : : processEvents ( ) ;
2018-04-28 05:33:26 +00:00
2020-08-08 18:17:54 +00:00
if ( date < firstImport ) {
2020-02-02 19:29:23 +00:00
# ifdef SESSION_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Skipping " < < date . toString ( ) < < " Before " < < firstImport . toString ( ) ;
2020-02-02 19:29:23 +00:00
# endif
2019-12-24 02:44:10 +00:00
continue ;
2018-04-28 05:33:26 +00:00
}
2020-08-08 18:17:54 +00:00
// This is not what we want to check, we must look at this day in the database files...
2021-07-08 18:27:22 +00:00
// Found the place in checkSummaryDay to compare session count with maskevents divided by 2
2020-08-13 23:32:34 +00:00
# ifdef SESSION_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " ResdayList size is " < < resdayList . size ( ) ;
2020-08-13 23:32:34 +00:00
# endif
2020-08-08 18:17:54 +00:00
// auto rit = resdayList.find(date);
// if (rit != resdayList.end()) {
// // Already seen this record.. should check if the data is the same, but meh.
// // At least check the maskeventcount to see if it changed...
// if ( maskeventcount->dataArray[0] != rit.value().str.maskevents ) {
// qDebug() << "Mask events don't match, purge" << rit.value().date.toString();
// // purge...
// }
// // #ifdef SESSION_DEBUG
// qDebug() << "Skipping" << date.toString() << "Already saw this one";
// // #endif
// continue;
// } // else {
// // qWarning() << date.toString() << "is missing from resdayList - FIX THIS";
// // continue;
// // }
2019-12-24 02:44:10 +00:00
int recstart = rec * maskon - > sampleCnt ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
bool validday = false ;
for ( int s = 0 ; s < maskon - > sampleCnt ; + + s ) {
qint32 on = maskon - > dataArray [ recstart + s ] ;
qint32 off = maskoff - > dataArray [ recstart + s ] ;
2018-04-28 05:33:26 +00:00
2020-11-19 20:51:01 +00:00
if ( ( ( on > = 0 ) & & ( off > = 0 ) ) & & ( on ! = off ) ) { // ignore very short on-off times
validday = true ;
}
2018-04-28 05:33:26 +00:00
}
2020-02-02 19:29:23 +00:00
if ( ! validday ) {
2019-12-24 02:44:10 +00:00
// There are no mask on/off events, so this STR day is useless.
qDebug ( ) < < " Skipping " < < date . toString ( ) < < " No mask events " ;
continue ;
2018-04-28 05:33:26 +00:00
}
2019-12-24 02:44:10 +00:00
2020-08-13 23:32:34 +00:00
# ifdef STR_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Adding " < < date . toString ( ) < < " to resdayLisyt b/c we have STR record " ;
2020-08-13 23:32:34 +00:00
# endif
2020-08-08 18:17:54 +00:00
auto rit = resdayList . insert ( date , ResMedDay ( date ) ) ;
2019-12-24 02:44:10 +00:00
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-02-06 14:13:16 +00:00
qDebug ( ) < < " Setting up STRRecord for " < < date . toString ( ) ;
2020-02-08 01:16:47 +00:00
# endif
2019-12-24 02:44:10 +00:00
STRRecord & R = rit . value ( ) . str ;
2020-02-10 04:45:31 +00:00
uint noonstamp = QDateTime ( date , QTime ( 12 , 0 , 0 ) , EDFInfo : : localNoDST ) . toTime_t ( ) ;
2019-12-24 02:44:10 +00:00
R . date = date ;
// skipday = false;
// For every mask on, there will be a session within 1 minute either way
// We can use that for data matching
// Scan the mask on/off events by minute
R . maskon . resize ( maskon - > sampleCnt ) ;
R . maskoff . resize ( maskoff - > sampleCnt ) ;
2020-02-06 14:13:16 +00:00
int lastOn = - 1 ;
int lastOff = - 1 ;
2019-12-24 02:44:10 +00:00
for ( int s = 0 ; s < maskon - > sampleCnt ; + + s ) {
qint32 on = maskon - > dataArray [ recstart + s ] ; // these on/off times are minutes since noon
2021-07-08 18:27:22 +00:00
qint32 off = maskoff - > dataArray [ recstart + s ] ; // we want the actual time in seconds
2020-11-19 20:51:01 +00:00
if ( ( on > 24 * 60 ) | | ( off > 24 * 60 ) ) {
qWarning ( ) . noquote ( ) < < " Mask times are out of range. Possible SDcard corruption " < < " date " < < date < < " on " < < on < < " off " < < off ;
continue ;
}
2020-02-06 14:13:16 +00:00
if ( on > 0 ) { // convert them to seconds since midnight
lastOn = s ;
R . maskon [ s ] = ( noonstamp + ( on * 60 ) ) ;
} else
R . maskon [ s ] = 0 ;
if ( off > 0 ) {
lastOff = s ;
R . maskoff [ s ] = ( noonstamp + ( off * 60 ) ) ;
} else
R . maskoff [ s ] = 0 ;
2018-04-28 05:33:26 +00:00
}
2019-12-24 02:44:10 +00:00
// two conditions that need dealing with, mask running at noon start, and finishing at noon start..
// (Sessions are forcibly split by resmed.. why the heck don't they store it that way???)
if ( ( R . maskon [ 0 ] = = 0 ) & & ( R . maskoff [ 0 ] > 0 ) ) {
R . maskon [ 0 ] = noonstamp ;
2020-02-08 01:16:47 +00:00
}
if ( ( lastOn > = 0 ) & & ( lastOff > = 0 ) ) {
if ( ( R . maskon [ lastOn ] > 0 ) & & ( R . maskoff [ lastOff ] = = 0 ) ) {
2020-02-10 04:45:31 +00:00
R . maskoff [ lastOff ] = QDateTime ( date , QTime ( 12 , 0 , 0 ) , EDFInfo : : localNoDST ) . addDays ( 1 ) . toTime_t ( ) - 1 ;
2020-02-08 01:16:47 +00:00
}
2018-04-28 05:33:26 +00:00
}
2020-02-06 14:13:16 +00:00
R . maskevents = maskeventcount - > dataArray [ rec ] ;
2019-12-24 02:44:10 +00:00
CPAPMode mode = MODE_UNKNOWN ;
2024-09-20 00:08:43 +00:00
IF ( AS_eleven )
DEBUGFC Q ( AS_eleven ) Q ( mode ) Q ( CPAP_Mode ) Q ( MODE_CPAP ) Q ( ( void * ) str . lookupSignal ( CPAP_Mode ) ) O ( " ___________________________________ " ) ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupSignal ( CPAP_Mode ) ) ) {
int mod = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
R . rms9_mode = mod ;
2024-09-20 00:08:43 +00:00
// Convert AirCurve 11 and AirSense 11 to appropiate resmed 10 modes.
DEBUGFC Q ( rec ) Q ( sig - > label ) Q ( sig - > transducer_type ) ;
DEBUGFC Q ( mod ) O ( " = " ) Q ( sig - > dataArray [ rec ] ) O ( " * " ) Q ( sig - > gain ) Q ( " + " ) Q ( sig - > offset ) ;
2021-10-28 00:26:18 +00:00
if ( AS_eleven ) { // translate AS11 mode values back to S9 / AS10 values
switch ( mod ) {
case 0 :
R . rms9_mode = 16 ; // Unknown
break ;
case 1 :
R . rms9_mode = 1 ; // still APAP
break ;
case 2 :
R . rms9_mode = 11 ; //make it look like A4Her
break ;
case 3 :
R . rms9_mode = 0 ; // make it be CPAP
break ;
2024-09-20 00:08:43 +00:00
case 7 : // Added for ASV .
case 8 : // Added for vAuto .
//case 9:
//case 10:
//case 11: // will 11 (A4Her) ever be generated by AS/AC 11?
break ; // no change these are the same as in aircurve 10
2021-10-28 00:26:18 +00:00
default :
R . rms9_mode = 16 ; // unknown for now
break ;
}
}
2018-04-28 05:33:26 +00:00
2022-01-03 21:23:11 +00:00
int RMS9_mode = R . rms9_mode ;
switch ( RMS9_mode ) {
case 11 :
2021-08-22 19:21:12 +00:00
mode = MODE_APAP ; // For her is a special apap
2022-01-03 21:23:11 +00:00
break ;
2022-05-06 03:18:51 +00:00
case 10 :
mode = MODE_UNKNOWN ; // it's PAC, whatever that is
break ;
2022-01-03 21:23:11 +00:00
case 9 :
2021-07-25 22:43:23 +00:00
mode = MODE_AVAPS ;
2022-01-03 21:23:11 +00:00
break ;
case 8 : // mod 8 == vpap adapt variable epap
2019-12-24 02:44:10 +00:00
mode = MODE_ASV_VARIABLE_EPAP ;
2022-01-03 21:23:11 +00:00
break ;
case 7 : // mod 7 == vpap adapt
2019-12-24 02:44:10 +00:00
mode = MODE_ASV ;
2022-01-03 21:23:11 +00:00
break ;
case 6 : // mod 6 == vpap auto (Min EPAP, Max IPAP, PS)
2019-12-24 02:44:10 +00:00
mode = MODE_BILEVEL_AUTO_FIXED_PS ;
2022-01-03 21:23:11 +00:00
break ;
case 5 : // 4,5 are S/T types...
case 4 :
case 3 : // mod 3 == vpap s fixed pressure (EPAP, IPAP, No PS)
mode = MODE_BILEVEL_FIXED ;
break ;
case 2 :
mode = MODE_BILEVEL_FIXED ;
break ;
case 1 :
2019-12-24 02:44:10 +00:00
mode = MODE_APAP ; // mod 1 == apap
2022-01-03 21:23:11 +00:00
break ;
case 0 :
2019-12-24 02:44:10 +00:00
mode = MODE_CPAP ; // mod 0 == cpap
2022-01-03 21:23:11 +00:00
break ;
default :
mode = MODE_UNKNOWN ;
}
2019-12-24 02:44:10 +00:00
R . mode = mode ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
// Settings.CPAP.Starting Pressure
2022-01-03 21:23:11 +00:00
if ( ( R . rms9_mode = = 0 ) & & ( sig = str . lookupLabel ( " S.C.StartPress " ) ) ) {
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2019-12-24 02:44:10 +00:00
}
2021-09-07 12:51:00 +00:00
// Settings.Adaptive Starting Pressure?
2022-01-03 21:23:11 +00:00
if ( ( R . rms9_mode = = 1 ) & & ( ( sig = str . lookupLabel ( " S.AS.StartPress " ) ) | | ( sig = str . lookupLabel ( " S.A.StartPress " ) ) ) ) {
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-08 18:27:22 +00:00
}
2021-09-07 12:51:00 +00:00
// mode 11 = APAP for her?
2022-01-03 21:23:11 +00:00
if ( ( R . rms9_mode = = 11 ) & & ( sig = str . lookupLabel ( " S.AFH.StartPress " ) ) ) {
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2019-12-24 02:44:10 +00:00
}
if ( ( R . mode = = MODE_BILEVEL_FIXED ) & & ( sig = str . lookupLabel ( " S.BL.StartPress " ) ) ) {
// Bilevel Starting Pressure
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2019-12-24 02:44:10 +00:00
}
2021-08-22 19:21:12 +00:00
if ( ( ( R . mode = = MODE_ASV ) | | ( R . mode = = MODE_ASV_VARIABLE_EPAP ) | |
( R . mode = = MODE_BILEVEL_AUTO_FIXED_PS ) ) & & ( sig = str . lookupLabel ( " S.VA.StartPress " ) ) ) {
2019-12-24 02:44:10 +00:00
// Bilevel Starting Pressure
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2019-12-24 02:44:10 +00:00
}
}
2021-07-25 22:43:23 +00:00
// Collect the staistics
2021-07-08 18:27:22 +00:00
if ( ( sig = str . lookupLabel ( " Mask Dur " ) ) | | ( sig = str . lookupLabel ( " Duration " ) ) ) {
2019-12-24 02:44:10 +00:00
R . maskdur = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " Leak Med " ) ) | | ( sig = str . lookupLabel ( " Leak.50 " ) ) ) {
float gain = sig - > gain * 60.0 ;
R . leak50 = EventDataType ( sig - > dataArray [ rec ] ) * gain ;
}
if ( ( sig = str . lookupLabel ( " Leak Max " ) ) | | ( sig = str . lookupLabel ( " Leak.Max " ) ) ) {
float gain = sig - > gain * 60.0 ;
R . leakmax = EventDataType ( sig - > dataArray [ rec ] ) * gain ;
}
if ( ( sig = str . lookupLabel ( " Leak 95 " ) ) | | ( sig = str . lookupLabel ( " Leak.95 " ) ) ) {
float gain = sig - > gain * 60.0 ;
R . leak95 = EventDataType ( sig - > dataArray [ rec ] ) * gain ;
}
if ( ( sig = str . lookupLabel ( " RespRate.50 " ) ) | | ( sig = str . lookupLabel ( " RR Med " ) ) ) {
R . rr50 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " RespRate.Max " ) ) | | ( sig = str . lookupLabel ( " RR Max " ) ) ) {
R . rrmax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " RespRate.95 " ) ) | | ( sig = str . lookupLabel ( " RR 95 " ) ) ) {
R . rr95 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " MinVent.50 " ) ) | | ( sig = str . lookupLabel ( " Min Vent Med " ) ) ) {
R . mv50 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " MinVent.Max " ) ) | | ( sig = str . lookupLabel ( " Min Vent Max " ) ) ) {
R . mvmax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " MinVent.95 " ) ) | | ( sig = str . lookupLabel ( " Min Vent 95 " ) ) ) {
R . mv95 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " TidVol.50 " ) ) | | ( sig = str . lookupLabel ( " Tid Vol Med " ) ) ) {
R . tv50 = EventDataType ( sig - > dataArray [ rec ] ) * ( sig - > gain * 1000.0 ) ;
}
if ( ( sig = str . lookupLabel ( " TidVol.Max " ) ) | | ( sig = str . lookupLabel ( " Tid Vol Max " ) ) ) {
R . tvmax = EventDataType ( sig - > dataArray [ rec ] ) * ( sig - > gain * 1000.0 ) ;
}
if ( ( sig = str . lookupLabel ( " TidVol.95 " ) ) | | ( sig = str . lookupLabel ( " Tid Vol 95 " ) ) ) {
R . tv95 = EventDataType ( sig - > dataArray [ rec ] ) * ( sig - > gain * 1000.0 ) ;
2018-04-28 05:33:26 +00:00
}
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " MaskPress.50 " ) ) | | ( sig = str . lookupLabel ( " Mask Pres Med " ) ) ) {
R . mp50 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " MaskPress.Max " ) ) | | ( sig = str . lookupLabel ( " Mask Pres Max " ) ) ) {
R . mpmax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " MaskPress.95 " ) ) | | ( sig = str . lookupLabel ( " Mask Pres 95 " ) ) ) {
R . mp95 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " TgtEPAP.50 " ) ) | | ( sig = str . lookupLabel ( " Exp Pres Med " ) ) ) {
R . tgtepap50 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " TgtEPAP.Max " ) ) | | ( sig = str . lookupLabel ( " Exp Pres Max " ) ) ) {
R . tgtepapmax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " TgtEPAP.95 " ) ) | | ( sig = str . lookupLabel ( " Exp Pres 95 " ) ) ) {
R . tgtepap95 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " TgtIPAP.50 " ) ) | | ( sig = str . lookupLabel ( " Insp Pres Med " ) ) ) {
R . tgtipap50 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " TgtIPAP.Max " ) ) | | ( sig = str . lookupLabel ( " Insp Pres Max " ) ) ) {
R . tgtipapmax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " TgtIPAP.95 " ) ) | | ( sig = str . lookupLabel ( " Insp Pres 95 " ) ) ) {
R . tgtipap95 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " I:E Med " ) ) ) {
R . ie50 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " I:E Max " ) ) ) {
R . iemax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
if ( ( sig = str . lookupLabel ( " I:E 95 " ) ) ) {
R . ie95 = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain ;
}
2018-04-28 05:33:26 +00:00
2021-07-25 22:43:23 +00:00
// Collect the pressure settings
2019-12-24 02:44:10 +00:00
bool haveipap = false ;
Q_UNUSED ( haveipap ) ;
2021-09-07 12:51:00 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupSignal ( CPAP_IPAP ) ) ) {
R . ipap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
haveipap = true ;
}
if ( ( sig = str . lookupSignal ( CPAP_EPAP ) ) ) {
R . epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2021-09-07 12:51:00 +00:00
if ( R . mode = = MODE_AVAPS ) {
if ( ( sig = str . lookupLabel ( " S.i.StartPress " ) ) ) {
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-09-07 12:51:00 +00:00
}
if ( ( sig = str . lookupLabel ( " S.i.EPAP " ) ) ) {
R . min_epap = R . max_epap = R . epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2022-04-22 16:33:32 +00:00
if ( ( sig = str . lookupLabel ( " S.i.EPAPAuto " ) ) ) {
R . epapAuto = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2021-09-07 12:51:00 +00:00
if ( ( sig = str . lookupLabel ( " S.i.MinPS " ) ) ) {
R . min_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.i.MinEPAP " ) ) ) {
R . min_epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.i.MaxEPAP " ) ) ) {
R . max_epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.i.MaxPS " ) ) ) {
R . max_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2022-04-22 16:33:32 +00:00
if ( ( R . epap > = 0 ) & & ( R . epapAuto = = 0 ) ) {
2021-09-07 12:51:00 +00:00
R . max_ipap = R . epap + R . max_ps ;
R . min_ipap = R . epap + R . min_ps ;
} else {
R . max_ipap = R . max_epap + R . max_ps ;
R . min_ipap = R . min_epap + R . min_ps ;
}
2022-04-22 16:33:32 +00:00
qDebug ( ) < < " AVAPS mode; Ramp " < < R . s_RampPressure < < " Fixed EPAP " < < ( ( R . epapAuto = = 0 ) ? " True " : " False " ) < <
" EPAP " < < R . epap < < " Min EPAP " < < R . min_epap < <
" Max EPAP " < < R . max_epap < < " Min PS " < < R . min_ps < < " Max PS " < < R . max_ps < <
" Min IPAP " < < R . min_ipap < < " Max_IPAP " < < R . max_ipap ;
2021-09-07 12:51:00 +00:00
}
2019-12-24 02:44:10 +00:00
if ( R . mode = = MODE_ASV ) {
if ( ( sig = str . lookupLabel ( " S.AV.StartPress " ) ) ) {
2022-04-22 16:33:32 +00:00
R . s_RampPressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.AV.EPAP " ) ) ) {
R . min_epap = R . max_epap = R . epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.AV.MinPS " ) ) ) {
R . min_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.AV.MaxPS " ) ) ) {
R . max_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
R . max_ipap = R . epap + R . max_ps ;
R . min_ipap = R . epap + R . min_ps ;
}
}
if ( R . mode = = MODE_ASV_VARIABLE_EPAP ) {
if ( ( sig = str . lookupLabel ( " S.AA.StartPress " ) ) ) {
EventDataType sp = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2022-04-22 16:33:32 +00:00
R . s_RampPressure = sp ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.AA.MinEPAP " ) ) ) {
R . min_epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.AA.MaxEPAP " ) ) ) {
R . max_epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.AA.MinPS " ) ) ) {
R . min_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.AA.MaxPS " ) ) ) {
R . max_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
R . max_ipap = R . max_epap + R . max_ps ;
R . min_ipap = R . min_epap + R . min_ps ;
}
}
2022-01-03 21:23:11 +00:00
if ( ( R . rms9_mode = = 11 ) & & ( sig = str . lookupLabel ( " S.AFH.MaxPress " ) ) ) {
2019-12-24 02:44:10 +00:00
R . max_pressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2022-01-03 21:23:11 +00:00
else if ( ( sig = str . lookupSignal ( CPAP_PressureMax ) ) ) {
R . max_pressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( R . rms9_mode = = 11 ) & & ( sig = str . lookupLabel ( " S.AFH.MinPress " ) ) ) {
R . min_pressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
else if ( ( sig = str . lookupSignal ( CPAP_PressureMin ) ) ) {
2019-12-24 02:44:10 +00:00
R . min_pressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupSignal ( RMS9_SetPressure ) ) ) {
R . set_pressure = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupSignal ( CPAP_EPAPHi ) ) ) {
R . max_epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupSignal ( CPAP_EPAPLo ) ) ) {
R . min_epap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupSignal ( CPAP_IPAPHi ) ) ) {
R . max_ipap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
haveipap = true ;
}
if ( ( sig = str . lookupSignal ( CPAP_IPAPLo ) ) ) {
R . min_ipap = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
haveipap = true ;
}
if ( ( sig = str . lookupSignal ( CPAP_PS ) ) ) {
R . ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
// Okay, problem here: THere are TWO PSMin & MAX dataArrays on the 36037 with the same string
// One is for ASV mode, and one is for ASVAuto
int psvar = ( mode = = MODE_ASV_VARIABLE_EPAP ) ? 1 : 0 ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " Max PS " , psvar ) ) ) {
R . max_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " Min PS " , psvar ) ) ) {
R . min_ps = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
// ///// if (!haveipap) {
// ///// }
2018-05-30 22:53:30 +00:00
2019-12-24 02:44:10 +00:00
if ( mode = = MODE_ASV_VARIABLE_EPAP ) {
R . min_ipap = R . min_epap + R . min_ps ;
R . max_ipap = R . max_epap + R . max_ps ;
} else if ( mode = = MODE_ASV ) {
R . min_ipap = R . epap + R . min_ps ;
R . max_ipap = R . epap + R . max_ps ;
}
2018-05-03 05:08:45 +00:00
2021-07-25 22:43:23 +00:00
// Collect the other settings
if ( ( sig = str . lookupLabel ( " S.AS.Comfort " ) ) ) {
R . s_Comfort = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
if ( AS_eleven )
R . s_Comfort - - ;
}
2019-12-24 02:44:10 +00:00
EventDataType epr = - 1 , epr_level = - 1 ;
2021-07-25 22:43:23 +00:00
bool a1x = false ; // AS-10 or AS-11
2021-08-22 19:21:12 +00:00
if ( ( mode = = MODE_CPAP ) | | ( mode = = MODE_APAP ) ) {
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupSignal ( RMS9_EPR ) ) ) {
epr = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
epr - - ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupSignal ( RMS9_EPRLevel ) ) ) {
epr_level = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.EPR.EPRType " ) ) ) {
2021-07-08 18:27:22 +00:00
a1x = true ;
2019-12-24 02:44:10 +00:00
epr = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
epr + = 1 ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
epr - - ;
2019-12-24 02:44:10 +00:00
}
int epr_on = 0 , clin_epr_on = 0 ;
2022-02-27 16:50:10 +00:00
if ( ( sig = str . lookupLabel ( " S.EPR.EPREnable " ) ) ) { // first check devices opinion
2021-07-08 18:27:22 +00:00
a1x = true ;
2019-12-24 02:44:10 +00:00
epr_on = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
epr_on - - ;
2019-12-24 02:44:10 +00:00
}
if ( epr_on & & ( sig = str . lookupLabel ( " S.EPR.ClinEnable " ) ) ) {
2021-07-08 18:27:22 +00:00
a1x = true ;
2019-12-24 02:44:10 +00:00
clin_epr_on = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
clin_epr_on - - ;
2019-12-24 02:44:10 +00:00
}
2021-07-08 18:27:22 +00:00
if ( a1x & & ! ( epr_on & & clin_epr_on ) ) {
2019-12-24 02:44:10 +00:00
epr = 0 ;
epr_level = 0 ;
}
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ( epr > = 0 ) & & ( epr_level > = 0 ) ) {
R . epr_level = epr_level ;
R . epr = epr ;
} else {
if ( epr > = 0 ) {
static bool warn = false ;
if ( ! warn ) { // just nag once
qDebug ( ) < < " If you can read this, please tell the developers you found a ResMed with EPR but no EPR_LEVEL so he can remove this warning " ;
2020-02-02 19:29:23 +00:00
// sleep(1);
2019-12-24 02:44:10 +00:00
warn = true ;
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
R . epr = ( epr > 0 ) ? 1 : 0 ;
R . epr_level = epr ;
} else if ( epr_level > = 0 ) {
R . epr_level = epr_level ;
R . epr = ( epr_level > 0 ) ? 1 : 0 ;
2018-05-03 05:08:45 +00:00
}
2018-04-28 05:33:26 +00:00
}
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " AHI " ) ) ) {
R . ahi = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " AI " ) ) ) {
R . ai = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " HI " ) ) ) {
R . hi = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " UAI " ) ) ) {
R . uai = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " CAI " ) ) ) {
R . cai = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " OAI " ) ) ) {
R . oai = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " CSR " ) ) ) {
R . csr = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2018-05-03 05:08:45 +00:00
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " S.RampTime " ) ) ) {
R . s_RampTime = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.RampEnable " ) ) ) {
R . s_RampEnable = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_RampEnable - - ;
2021-09-07 12:51:00 +00:00
if ( R . s_RampEnable = = 2 )
R . s_RampTime = - 1 ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.EPR.ClinEnable " ) ) ) {
R . s_EPR_ClinEnable = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_EPR_ClinEnable - - ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.EPR.EPREnable " ) ) ) {
R . s_EPREnable = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_EPREnable - - ;
2019-12-24 02:44:10 +00:00
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " S.ABFilter " ) ) ) {
R . s_ABFilter = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_ABFilter - - ;
2019-12-24 02:44:10 +00:00
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " S.ClimateControl " ) ) ) {
R . s_ClimateControl = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_ClimateControl - - ;
2019-12-24 02:44:10 +00:00
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ( sig = str . lookupLabel ( " S.Mask " ) ) ) {
R . s_Mask = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-09-07 12:51:00 +00:00
if ( AS_eleven ) {
if ( R . s_Mask < 2 | | R . s_Mask > 4 )
R . s_Mask = 4 ; // unknown mask type
else
R . s_Mask - = 2 ; // why be consistent?
}
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.PtAccess " ) ) ) {
2021-08-17 15:32:39 +00:00
if ( AS_eleven ) {
R . s_PtView = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
R . s_PtView - - ;
} else
R . s_PtAccess = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.SmartStart " ) ) ) {
R . s_SmartStart = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_SmartStart - - ;
2021-08-22 21:39:13 +00:00
// qDebug() << "SmartStart is set to" << R.s_SmartStart;
2021-07-25 22:43:23 +00:00
}
if ( ( sig = str . lookupLabel ( " S.SmartStop " ) ) ) {
R . s_SmartStop = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
if ( AS_eleven )
R . s_SmartStop - - ;
2024-09-20 00:08:43 +00:00
//qDebug() << "SmartStop is set to" << R.s_SmartStop;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.HumEnable " ) ) ) {
R . s_HumEnable = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_HumEnable - - ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.HumLevel " ) ) ) {
R . s_HumLevel = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.TempEnable " ) ) ) {
R . s_TempEnable = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2021-07-25 22:43:23 +00:00
if ( AS_eleven )
R . s_TempEnable - - ;
2019-12-24 02:44:10 +00:00
}
if ( ( sig = str . lookupLabel ( " S.Temp " ) ) ) {
R . s_Temp = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.Tube " ) ) ) {
R . s_Tube = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2022-04-22 16:33:32 +00:00
}
2022-05-06 03:18:51 +00:00
if ( ( R . rms9_mode > = 2 ) & & ( R . rms9_mode < = 5 ) ) { // S, ST, or T modes
2024-09-20 00:08:43 +00:00
//qDebug() << "BiLevel Mode found" << R.rms9_mode;
2022-05-08 21:41:53 +00:00
if ( R . rms9_mode = = 3 ) { // S mode only
2022-05-06 03:18:51 +00:00
if ( ( sig = str . lookupLabel ( " S.EasyBreathe " ) ) ) {
R . s_EasyBreathe = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
}
if ( ( sig = str . lookupLabel ( " S.RiseEnable " ) ) ) {
R . s_RiseEnable = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.RiseTime " ) ) ) {
R . s_RiseTime = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( R . rms9_mode = = 3 ) | | ( R . rms9_mode = = 4 ) ) { // S or ST mode
if ( ( sig = str . lookupLabel ( " S.Cycle " ) ) ) {
R . s_Cycle = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.Trigger " ) ) ) {
R . s_Trigger = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.TiMax " ) ) ) {
R . s_TiMax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
if ( ( sig = str . lookupLabel ( " S.TiMin " ) ) ) {
R . s_TiMin = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
}
}
}
if ( R . rms9_mode = = 6 ) { // vAuto mode
2022-11-12 19:46:57 +00:00
qDebug ( ) < < " vAuto mode found " < < 6 ;
2022-05-06 03:18:51 +00:00
if ( ( sig = str . lookupLabel ( " S.Cycle " ) ) ) {
2022-05-08 21:41:53 +00:00
R . s_Cycle = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2024-09-20 00:08:43 +00:00
// qDebug() << "Cycle" << R.s_Cycle;
2022-05-06 03:18:51 +00:00
}
if ( ( sig = str . lookupLabel ( " S.Trigger " ) ) ) {
R . s_Trigger = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2024-09-20 00:08:43 +00:00
// qDebug() << "Trigger" << R.s_Trigger;
2022-05-06 03:18:51 +00:00
}
if ( ( sig = str . lookupLabel ( " S.TiMax " ) ) ) {
R . s_TiMax = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2024-09-20 00:08:43 +00:00
// qDebug() << QString("TiMax %1").arg( R.s_TiMax, 0, 'f', 1);
2022-05-06 03:18:51 +00:00
}
if ( ( sig = str . lookupLabel ( " S.TiMin " ) ) ) {
R . s_TiMin = EventDataType ( sig - > dataArray [ rec ] ) * sig - > gain + sig - > offset ;
2024-09-20 00:08:43 +00:00
// qDebug() << QString("TiMin %1").arg( R.s_TiMin, 0, 'f', 1);
2022-05-06 03:18:51 +00:00
}
}
2022-01-08 00:25:56 +00:00
if ( R . min_pressure = = 0 ) {
qDebug ( ) < < " Min Pressure is zero on " < < date . toString ( ) ;
}
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-02-06 14:13:16 +00:00
qDebug ( ) < < " Finished " < < date . toString ( ) ;
2020-02-08 01:16:47 +00:00
# endif
2019-12-24 02:44:10 +00:00
}
2020-04-29 17:57:10 +00:00
# ifdef STR_DEBUG
qDebug ( ) < < " Finished " < < strfile ;
# endif
2019-12-24 02:44:10 +00:00
}
2020-02-06 14:13:16 +00:00
qDebug ( ) < < " Finished ProcessSTR " ;
2020-02-14 20:18:25 +00:00
return true ;
2018-04-28 05:33:26 +00:00
}
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
2022-02-27 16:50:10 +00:00
// Parse Identification.tgt file (containing serial number and device information)
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
2021-06-07 02:57:52 +00:00
// QHash<QString, QString> parseIdentLine( const QString line, MachineInfo * info); // forward
// void scanProductObject( QJsonObject product, MachineInfo *info, QHash<QString, QString> *idmap); // forward
bool parseIdentFile ( QString path , MachineInfo * info , QHash < QString , QString > & idmap ) {
2019-07-22 21:01:47 +00:00
QString filename = path + RMS9_STR_idfile + STR_ext_TGT ;
2011-12-30 23:02:45 +00:00
QFile f ( filename ) ;
2021-06-07 02:57:52 +00:00
QFile j ( path + RMS9_STR_idfile + STR_ext_JSON ) ;
2014-04-17 05:58:57 +00:00
2021-10-28 03:23:38 +00:00
if ( j . exists ( ) ) { // chose the AS11 file if both exist
2021-06-07 02:57:52 +00:00
if ( ! j . open ( QIODevice : : ReadOnly ) ) {
return false ;
}
QByteArray identData = j . readAll ( ) ;
j . close ( ) ;
QJsonDocument identDoc ( QJsonDocument : : fromJson ( identData ) ) ;
QJsonObject identObj ( identDoc . object ( ) ) ;
if ( identObj . contains ( " FlowGenerator " ) & & identObj [ " FlowGenerator " ] . isObject ( ) ) {
QJsonObject flow = identObj [ " FlowGenerator " ] . toObject ( ) ;
if ( flow . contains ( " IdentificationProfiles " ) & & flow [ " IdentificationProfiles " ] . isObject ( ) ) {
QJsonObject profiles = flow [ " IdentificationProfiles " ] . toObject ( ) ;
if ( profiles . contains ( " Product " ) & & profiles [ " Product " ] . isObject ( ) ) {
QJsonObject product = profiles [ " Product " ] . toObject ( ) ;
2021-10-28 03:23:38 +00:00
// passed in MachineInfo info = newInfo();
2021-06-07 02:57:52 +00:00
scanProductObject ( product , info , & idmap ) ;
return true ;
}
}
}
2021-10-28 03:23:38 +00:00
return false ;
}
// Abort if this file is dodgy..
if ( f . exists ( ) ) {
if ( ! f . open ( QIODevice : : ReadOnly ) ) {
return false ;
}
qDebug ( ) < < " Parsing Identification File " < < filename ;
// emit updateMessage(QObject::tr("Parsing Identification File"));
// QApplication::processEvents();
// Parse # entries into idmap.
while ( ! f . atEnd ( ) ) {
QString line = f . readLine ( ) . trimmed ( ) ;
QHash < QString , QString > hash = parseIdentLine ( line , info ) ;
2023-02-18 13:58:47 +00:00
idmap . QTCOMBINE ( hash ) ;
2021-10-28 03:23:38 +00:00
}
2021-06-07 02:57:52 +00:00
2021-10-28 03:23:38 +00:00
f . close ( ) ;
return true ;
2011-12-30 23:02:45 +00:00
}
2021-06-07 02:57:52 +00:00
return false ;
}
2014-04-17 05:58:57 +00:00
2021-06-07 02:57:52 +00:00
void scanProductObject ( QJsonObject product , MachineInfo * info , QHash < QString , QString > * idmap ) {
QHash < QString , QString > hash1 , hash2 , hash3 ;
if ( product . contains ( " SerialNumber " ) ) {
info - > serial = product [ " SerialNumber " ] . toString ( ) ;
hash1 [ " SerialNumber " ] = product [ " SerialNumber " ] . toString ( ) ;
if ( idmap )
2023-02-18 13:58:47 +00:00
idmap - > QTCOMBINE ( hash1 ) ;
2021-06-07 02:57:52 +00:00
}
if ( product . contains ( " ProductCode " ) ) {
info - > modelnumber = product [ " ProductCode " ] . toString ( ) ;
hash2 [ " ProductCode " ] = info - > modelnumber ;
if ( idmap )
2023-02-18 13:58:47 +00:00
idmap - > QTCOMBINE ( hash2 ) ;
2021-06-07 02:57:52 +00:00
}
if ( product . contains ( " ProductName " ) ) {
info - > model = product [ " ProductName " ] . toString ( ) ;
hash3 [ " ProductName " ] = info - > model ;
if ( idmap )
2023-02-18 13:58:47 +00:00
idmap - > QTCOMBINE ( hash3 ) ;
2021-06-07 02:57:52 +00:00
int idx = info - > model . indexOf ( " 11 " ) ;
info - > series = info - > model . left ( idx + 2 ) ;
}
2019-07-22 21:01:47 +00:00
}
2020-05-17 23:24:31 +00:00
void backupSTRfiles ( const QString strpath , const QString importPath , const QString backupPath ,
2020-02-08 01:16:47 +00:00
MachineInfo & info , QMap < QDate , STRFile > & STRmap )
{
2020-05-18 21:42:37 +00:00
Q_UNUSED ( strpath ) ;
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Starting backupSTRfiles during new IMPORT " ;
2020-05-15 23:58:11 +00:00
QDir dir ;
2020-05-17 23:24:31 +00:00
// Qstring strBackupPath(backupPath+"STR_Backup");
2019-08-16 22:05:25 +00:00
QStringList strfiles ;
2020-05-18 21:42:37 +00:00
// add Backup/STR.edf - make sure it ends up in the STRmap
strfiles . push_back ( backupPath + " STR.edf " ) ;
2019-08-16 22:05:25 +00:00
2020-05-17 23:24:31 +00:00
// Just in case we are importing from a Backup folder in a different Profile, process OSCAR backup structures
QString strBackupPath ( importPath + " STR_Backup " ) ;
2020-05-15 23:58:11 +00:00
dir . setPath ( strBackupPath ) ;
2019-08-16 22:05:25 +00:00
dir . setFilter ( QDir : : Files | QDir : : Hidden | QDir : : Readable ) ;
QFileInfoList flist = dir . entryInfoList ( ) ;
// Add any STR_Backup versions to the file list
2020-05-17 23:24:31 +00:00
for ( auto & fi : flist ) { // this is empty if imprting from an SD card
2019-08-16 22:05:25 +00:00
QString filename = fi . fileName ( ) ;
if ( ! filename . startsWith ( " STR " , Qt : : CaseInsensitive ) )
continue ;
if ( ! ( filename . endsWith ( " edf.gz " , Qt : : CaseInsensitive ) | | filename . endsWith ( " edf " , Qt : : CaseInsensitive ) ) )
continue ;
strfiles . push_back ( fi . canonicalFilePath ( ) ) ;
}
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
qDebug ( ) < < " STR file list size is " < < strfiles . size ( ) ;
# endif
2019-08-16 22:05:25 +00:00
2020-05-16 20:16:10 +00:00
// Now copy any of these files to the Backup folder adding the file date to the file name
// and put it into the STRmap structure
2019-08-16 22:05:25 +00:00
for ( auto & filename : strfiles ) {
2020-05-17 23:24:31 +00:00
QDate date ;
long int days ;
ResMedEDFInfo * stredf = fetchSTRandVerify ( filename , info . serial ) ;
if ( stredf = = nullptr )
2019-08-16 22:05:25 +00:00
continue ;
2020-05-17 23:24:31 +00:00
date = stredf - > edfHdr . startdate_orig . date ( ) ;
days = stredf - > GetNumDataRecords ( ) ;
2019-08-16 22:05:25 +00:00
if ( STRmap . contains ( date ) ) {
2020-05-17 23:24:31 +00:00
qDebug ( ) < < " STRmap already contains " < < date . toString ( " yyyy-MM-dd " ) < < " for " < < STRmap [ date ] . days < < " ending " < < date . addDays ( STRmap [ date ] . days - 1 ) ;
qDebug ( ) < < filename . section ( " / " , - 2 , - 1 ) < < " has " < < days < < " ending " < < date . addDays ( days - 1 ) ;
2020-04-27 00:59:23 +00:00
if ( days < = STRmap [ date ] . days ) {
2020-05-16 21:31:18 +00:00
qDebug ( ) < < " Skipping " < < filename . section ( " / " , - 2 , - 1 ) < < " Keeping " < < STRmap [ date ] . filename . section ( " / " , - 2 , - 1 ) ;
2020-04-27 00:59:23 +00:00
delete stredf ;
continue ;
2020-05-17 23:24:31 +00:00
} else {
qDebug ( ) < < " Dropping " < < STRmap [ date ] . filename . section ( " / " , - 2 , - 1 ) < < " Keeping " < < filename . section ( " / " , - 2 , - 1 ) ;
delete STRmap [ date ] . edf ;
STRmap . remove ( date ) ; // new one gets added after we know its new name
2020-04-27 00:59:23 +00:00
}
2019-08-16 22:05:25 +00:00
}
2020-05-17 23:24:31 +00:00
// now create the new backup name
2020-05-16 20:16:10 +00:00
QString newname = " STR- " + date . toString ( " yyyyMMdd " ) + " . " + STR_ext_EDF ;
2020-05-17 23:24:31 +00:00
QString backupfile = backupPath + " /STR_Backup/ " + newname ;
2019-08-16 22:05:25 +00:00
QString gzfile = backupfile + STR_ext_gz ;
QString nongzfile = backupfile ;
bool compress_backups = p_profile - > session - > compressBackupData ( ) ;
backupfile = compress_backups ? gzfile : nongzfile ;
2020-05-16 20:16:10 +00:00
STRmap [ date ] = STRFile ( backupfile , days , stredf ) ;
2020-05-19 18:47:54 +00:00
qDebug ( ) < < " Adding " < < filename . section ( " / " , - 3 , - 1 ) < < " with " < < days < < " days as " < < backupfile . section ( " / " , - 3 , - 1 ) < < " to STRmap " ;
2020-05-16 20:16:10 +00:00
2020-05-17 23:24:31 +00:00
if ( QFile : : exists ( backupfile ) ) {
QFile : : remove ( backupfile ) ;
}
2020-08-08 18:17:54 +00:00
# ifdef STR_DEBUG
qDebug ( ) < < " Copying " < < filename . section ( " / " , - 3 , - 1 ) < < " to " < < backupfile . section ( " / " , - 3 , - 1 ) ;
# endif
2020-05-17 23:24:31 +00:00
if ( filename . endsWith ( STR_ext_gz , Qt : : CaseInsensitive ) ) { // we have a compressed file
if ( compress_backups ) { // fine, copy it to backup folder
2020-08-13 23:32:34 +00:00
if ( ! QFile : : copy ( filename , backupfile ) )
qWarning ( ) < < " Failed to copy " < < filename < < " to " < < backupfile ;
2020-05-17 23:24:31 +00:00
} else { // oops, uncompress it to the backup folder
uncompressFile ( filename , backupfile ) ;
}
} else { // file is not compressed
if ( compress_backups ) { // so compress it into the backup folder
compressFile ( filename , backupfile ) ;
} else { // and that's OK, just copy it over
2020-08-13 23:32:34 +00:00
if ( ! QFile : : copy ( filename , backupfile ) )
qWarning ( ) < < " Failed to copy " < < filename < < " to " < < backupfile ;
2019-08-16 22:05:25 +00:00
}
}
2020-05-16 20:16:10 +00:00
2019-08-16 22:05:25 +00:00
// Remove any duplicate compressed/uncompressed backup file
if ( compress_backups )
QFile : : exists ( nongzfile ) & & QFile : : remove ( nongzfile ) ;
else
QFile : : exists ( gzfile ) & & QFile : : remove ( gzfile ) ;
} // end for walking the STR files list
2020-02-08 01:16:47 +00:00
# ifdef STR_DEBUG
2020-02-06 22:17:08 +00:00
qDebug ( ) < < " STRmap has " < < STRmap . size ( ) < < " entries " ;
2020-02-08 01:16:47 +00:00
# endif
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Finished backupSTRfiles during new IMPORT " ;
2019-08-16 22:05:25 +00:00
}
2019-12-24 02:44:10 +00:00
QHash < QString , QString > parseIdentLine ( const QString line , MachineInfo * info )
{
QHash < QString , QString > hash ;
if ( ! line . isEmpty ( ) ) {
QString key = line . section ( " " , 0 , 0 ) . section ( " # " , 1 ) ;
QString value = line . section ( " " , 1 ) ;
2019-08-16 23:50:09 +00:00
2019-12-24 02:44:10 +00:00
if ( key = = " SRN " ) { // Serial Number
info - > serial = value ;
2019-08-16 23:50:09 +00:00
2019-12-24 02:44:10 +00:00
} else if ( key = = " PNA " ) { // Product Name
value . replace ( " _ " , " " ) ;
2019-08-16 23:50:09 +00:00
2020-03-23 18:41:31 +00:00
if ( value . contains ( STR_ResMed_AirSense10 ) ) {
// value.replace(STR_ResMed_AirSense10, "");
info - > series = STR_ResMed_AirSense10 ;
} else if ( value . contains ( STR_ResMed_AirCurve10 ) ) {
// value.replace(STR_ResMed_AirCurve10, "");
info - > series = STR_ResMed_AirCurve10 ;
} else { // it will be a Series 9, and might not contain (STR_ResMed_S9))
value . replace ( " ( " , " " ) ; // might sometimes have a double space...
2020-03-19 21:54:48 +00:00
value . replace ( " ) " , " " ) ;
if ( ! value . startsWith ( STR_ResMed_S9 ) ) {
value . replace ( STR_ResMed_S9 , " " ) ;
value . insert ( 0 , " " ) ; // There's proablely a better way than this
value . insert ( 0 , STR_ResMed_S9 ) ; // two step way to put "S9 " at the start
}
info - > series = STR_ResMed_S9 ;
// value.replace(STR_ResMed_S9, "");
2019-12-24 02:44:10 +00:00
}
// if (value.contains("Adapt", Qt::CaseInsensitive)) {
// if (!value.contains("VPAP")) {
// value.replace("Adapt", QObject::tr("VPAP Adapt"));
// }
// }
info - > model = value . trimmed ( ) ;
} else if ( key = = " PCD " ) { // Product Code
info - > modelnumber = value ;
2019-08-16 23:50:09 +00:00
}
2019-12-24 02:44:10 +00:00
hash [ key ] = value ;
}
2019-08-16 23:50:09 +00:00
2019-12-24 02:44:10 +00:00
return hash ;
2019-08-16 23:50:09 +00:00
}
2019-12-24 02:44:10 +00:00
EDFType lookupEDFType ( const QString & filename )
2019-07-22 21:01:47 +00:00
{
2019-12-24 02:44:10 +00:00
QString text = filename . section ( " _ " , - 1 ) . section ( " . " , 0 , 0 ) . toUpper ( ) ;
if ( text = = " EVE " ) {
return EDF_EVE ;
} else if ( text = = " BRP " ) {
return EDF_BRP ;
} else if ( text = = " PLD " ) {
return EDF_PLD ;
2023-10-14 02:53:39 +00:00
} else if ( ( text = = " SAD " ) | | ( text = = " SA2 " ) ) {
2019-12-24 02:44:10 +00:00
return EDF_SAD ;
} else if ( text = = " CSL " ) {
return EDF_CSL ;
2020-03-23 18:41:31 +00:00
} else if ( text = = " AEV " ) {
return EDF_AEV ;
2019-12-24 02:44:10 +00:00
} else return EDF_UNKNOWN ;
}
2019-07-22 21:01:47 +00:00
2019-12-24 02:44:10 +00:00
///////////////////////////////////////////////////////////////////////////////
// Looks inside an EDF or EDF.gz and grabs the start and duration
///////////////////////////////////////////////////////////////////////////////
EDFduration getEDFDuration ( const QString & filename )
{
2019-12-30 16:14:05 +00:00
// qDebug() << "getEDFDuration called for" << filename;
2019-07-22 21:01:47 +00:00
2019-12-24 02:44:10 +00:00
QString ext = filename . section ( " _ " , - 1 ) . section ( " . " , 0 , 0 ) . toUpper ( ) ;
2019-07-22 21:01:47 +00:00
2019-12-24 02:44:10 +00:00
if ( ( ext = = " EVE " ) | | ( ext = = " CSL " ) ) { // don't even try with Annotation-only edf files
EDFduration dur ( 0 , 0 , filename ) ;
dur . type = lookupEDFType ( filename ) ;
qDebug ( ) < < " File ext is " < < ext ;
dumpEDFduration ( dur ) ;
return dur ;
2019-07-22 21:01:47 +00:00
}
2019-12-24 02:44:10 +00:00
bool ok1 , ok2 ;
int num_records ;
double rec_duration ;
QDateTime startDate ;
2019-08-28 03:59:51 +00:00
2020-02-02 19:29:23 +00:00
// We will just look at the header part of the edf file here
2019-12-24 02:44:10 +00:00
if ( ! filename . endsWith ( " .gz " , Qt : : CaseInsensitive ) ) {
QFile file ( filename ) ;
if ( ! file . open ( QFile : : ReadOnly ) )
return EDFduration ( 0 , 0 , filename ) ;
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ! file . seek ( 0xa8 ) ) {
file . close ( ) ;
return EDFduration ( 0 , 0 , filename ) ;
}
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
QByteArray bytes = file . read ( 16 ) . trimmed ( ) ;
2020-02-02 19:29:23 +00:00
// We'll fix the xx85 problem below
// startDate = QDateTime::fromString(QString::fromLatin1(bytes, 16), "dd.MM.yyHH.mm.ss");
2020-02-10 04:45:31 +00:00
// getStartDT ought to be named getStartNoDST ... TODO someday
2020-02-02 19:29:23 +00:00
startDate = EDFInfo : : getStartDT ( QString : : fromLatin1 ( bytes , 16 ) ) ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
if ( ! file . seek ( 0xec ) ) {
file . close ( ) ;
return EDFduration ( 0 , 0 , filename ) ;
2012-01-05 04:37:22 +00:00
}
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
bytes = file . read ( 8 ) . trimmed ( ) ;
num_records = bytes . toInt ( & ok1 ) ;
bytes = file . read ( 8 ) . trimmed ( ) ;
rec_duration = bytes . toDouble ( & ok2 ) ;
2019-08-23 13:56:17 +00:00
2019-12-24 02:44:10 +00:00
file . close ( ) ;
} else {
gzFile f = gzopen ( filename . toLatin1 ( ) , " rb " ) ;
if ( ! f )
return EDFduration ( 0 , 0 , filename ) ;
2019-08-23 13:56:17 +00:00
2019-12-24 02:44:10 +00:00
// Decompressed header and data block
if ( ! gzseek ( f , 0xa8 , SEEK_SET ) ) {
gzclose ( f ) ;
return EDFduration ( 0 , 0 , filename ) ;
}
char datebytes [ 17 ] = { 0 } ;
gzread ( f , ( char * ) & datebytes , 16 ) ;
QString str = QString ( QString : : fromLatin1 ( datebytes , 16 ) ) . trimmed ( ) ;
2020-02-02 19:29:23 +00:00
// startDate = QDateTime::fromString(str, "dd.MM.yyHH.mm.ss");
startDate = EDFInfo : : getStartDT ( str ) ;
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
if ( ! gzseek ( f , 0xec - 0xa8 - 16 , SEEK_CUR ) ) { // 0xec
gzclose ( f ) ;
return EDFduration ( 0 , 0 , filename ) ;
}
2012-01-05 04:37:22 +00:00
2019-12-24 02:44:10 +00:00
char cbytes [ 9 ] = { 0 } ;
gzread ( f , ( char * ) & cbytes , 8 ) ;
str = QString ( cbytes ) . trimmed ( ) ;
num_records = str . toInt ( & ok1 ) ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
gzread ( f , ( char * ) & cbytes , 8 ) ;
str = QString ( cbytes ) . trimmed ( ) ;
rec_duration = str . toDouble ( & ok2 ) ;
2012-01-05 04:37:22 +00:00
2019-12-24 02:44:10 +00:00
gzclose ( f ) ;
2011-06-29 19:06:49 +00:00
}
2011-09-17 12:39:00 +00:00
2019-12-24 02:44:10 +00:00
QDate d2 = startDate . date ( ) ;
if ( d2 . year ( ) < 2000 ) {
d2 . setDate ( d2 . year ( ) + 100 , d2 . month ( ) , d2 . day ( ) ) ;
startDate . setDate ( d2 ) ;
}
2020-05-07 20:43:52 +00:00
if ( ( ! startDate . isValid ( ) ) | | ( startDate > QDateTime : : currentDateTime ( ) ) ) {
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Invalid date time retreieved parsing EDF duration for " < < filename ;
2020-02-08 01:16:47 +00:00
qDebug ( ) < < " Time zone(Utc) is " < < startDate . timeZone ( ) . abbreviation ( QDateTime : : currentDateTimeUtc ( ) ) ;
qDebug ( ) < < " Time zone is " < < startDate . timeZone ( ) . abbreviation ( QDateTime : : currentDateTime ( ) ) ;
2019-12-24 02:44:10 +00:00
return EDFduration ( 0 , 0 , filename ) ;
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
if ( ! ( ok1 & & ok2 ) )
return EDFduration ( 0 , 0 , filename ) ;
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
quint32 start = startDate . toTime_t ( ) ;
quint32 end = start + rec_duration * num_records ;
2018-05-07 00:37:22 +00:00
2019-12-24 02:44:10 +00:00
QString filedate = filename . section ( " / " , - 1 ) . section ( " _ " , 0 , 1 ) ;
2020-02-02 19:29:23 +00:00
// QDateTime dt2 = QDateTime::fromString(filedate, "yyyyMMdd_hhmmss");
d2 = QDate : : fromString ( filedate . left ( 8 ) , " yyyyMMdd " ) ;
QTime t2 = QTime : : fromString ( filedate . right ( 6 ) , " hhmmss " ) ;
QDateTime dt2 = QDateTime ( d2 , t2 , EDFInfo : : localNoDST ) ;
2019-12-24 02:44:10 +00:00
quint32 st2 = dt2 . toTime_t ( ) ;
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
start = qMin ( st2 , start ) ; // They should be the same, usually
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( end < start )
end = qMax ( st2 , start ) ;
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
EDFduration dur ( start , end , filename ) ;
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
dur . type = lookupEDFType ( filename ) ;
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
return dur ;
}
2018-04-25 18:04:05 +00:00
2014-05-18 17:06:58 +00:00
2019-12-24 02:44:10 +00:00
void GuessPAPMode ( Session * sess )
{
if ( sess - > channelDataExists ( CPAP_Pressure ) ) {
// Determine CPAP or APAP?
EventDataType min = sess - > Min ( CPAP_Pressure ) ;
EventDataType max = sess - > Max ( CPAP_Pressure ) ;
if ( ( max - min ) < 0.1 ) {
sess - > settings [ CPAP_Mode ] = MODE_CPAP ;
sess - > settings [ CPAP_Pressure ] = qRound ( max * 10.0 ) / 10.0 ;
// early call.. It's CPAP mode
} else {
// Ramp is ugly - but this is a bad way to test for it
if ( sess - > length ( ) > 1800000L ) { // half an hour
}
sess - > settings [ CPAP_Mode ] = MODE_APAP ;
sess - > settings [ CPAP_PressureMin ] = qRound ( min * 10.0 ) / 10.0 ;
sess - > settings [ CPAP_PressureMax ] = qRound ( max * 10.0 ) / 10.0 ;
2018-05-03 05:08:45 +00:00
}
2019-12-24 02:44:10 +00:00
} else if ( sess - > eventlist . contains ( CPAP_IPAP ) ) {
sess - > settings [ CPAP_Mode ] = MODE_BILEVEL_AUTO_VARIABLE_PS ;
// Determine BiPAP or ASV
}
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
}
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
void StoreSummaryStatistics ( Session * sess , STRRecord & R )
{
if ( R . mode > = 0 ) {
if ( R . mode = = MODE_CPAP ) {
} else if ( R . mode = = MODE_APAP ) {
}
}
2018-05-07 00:37:22 +00:00
2019-12-24 02:44:10 +00:00
if ( R . leak50 > = 0 ) {
2020-02-02 19:29:23 +00:00
// sess->setp95(CPAP_Leak, R.leak95);
// sess->setp50(CPAP_Leak, R.leak50);
2019-12-24 02:44:10 +00:00
sess - > setMax ( CPAP_Leak , R . leakmax ) ;
}
2018-05-03 05:08:45 +00:00
2019-12-24 02:44:10 +00:00
if ( R . rr50 > = 0 ) {
2020-02-02 19:29:23 +00:00
// sess->setp95(CPAP_RespRate, R.rr95);
// sess->setp50(CPAP_RespRate, R.rr50);
2019-12-24 02:44:10 +00:00
sess - > setMax ( CPAP_RespRate , R . rrmax ) ;
2018-05-03 05:08:45 +00:00
}
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
if ( R . mv50 > = 0 ) {
2020-02-02 19:29:23 +00:00
// sess->setp95(CPAP_MinuteVent, R.mv95);
// sess->setp50(CPAP_MinuteVent, R.mv50);
2019-12-24 02:44:10 +00:00
sess - > setMax ( CPAP_MinuteVent , R . mvmax ) ;
}
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
if ( R . tv50 > = 0 ) {
2020-02-02 19:29:23 +00:00
// sess->setp95(CPAP_TidalVolume, R.tv95);
// sess->setp50(CPAP_TidalVolume, R.tv50);
2019-12-24 02:44:10 +00:00
sess - > setMax ( CPAP_TidalVolume , R . tvmax ) ;
}
2018-05-03 09:59:31 +00:00
2019-12-24 02:44:10 +00:00
if ( R . mp50 > = 0 ) {
2020-02-02 19:29:23 +00:00
// sess->setp95(CPAP_MaskPressure, R.mp95);
// sess->seTTtp50(CPAP_MaskPressure, R.mp50);
2019-12-24 02:44:10 +00:00
sess - > setMax ( CPAP_MaskPressure , R . mpmax ) ;
}
2012-01-05 04:37:22 +00:00
2019-12-24 02:44:10 +00:00
if ( R . oai > 0 ) {
sess - > setCph ( CPAP_Obstructive , R . oai ) ;
sess - > setCount ( CPAP_Obstructive , R . oai * sess - > hours ( ) ) ;
}
if ( R . hi > 0 ) {
sess - > setCph ( CPAP_Hypopnea , R . hi ) ;
sess - > setCount ( CPAP_Hypopnea , R . hi * sess - > hours ( ) ) ;
2012-01-05 04:37:22 +00:00
}
2019-12-24 02:44:10 +00:00
if ( R . cai > 0 ) {
sess - > setCph ( CPAP_ClearAirway , R . cai ) ;
sess - > setCount ( CPAP_ClearAirway , R . cai * sess - > hours ( ) ) ;
}
if ( R . uai > 0 ) {
sess - > setCph ( CPAP_Apnea , R . uai ) ;
sess - > setCount ( CPAP_Apnea , R . uai * sess - > hours ( ) ) ;
}
if ( R . csr > 0 ) {
sess - > setCph ( CPAP_CSR , R . csr ) ;
sess - > setCount ( CPAP_CSR , R . csr * sess - > hours ( ) ) ;
}
}
2011-12-30 23:02:45 +00:00
2019-12-24 02:44:10 +00:00
void StoreSettings ( Session * sess , STRRecord & R )
{
if ( R . mode > = 0 ) {
sess - > settings [ CPAP_Mode ] = R . mode ;
sess - > settings [ RMS9_Mode ] = R . rms9_mode ;
2022-01-02 01:49:40 +00:00
if ( R . min_pressure = = 0 )
qDebug ( ) < < " Min Pressure is zero, R.mode is " < < R . mode ;
2019-12-24 02:44:10 +00:00
if ( R . mode = = MODE_CPAP ) {
if ( R . set_pressure > = 0 ) sess - > settings [ CPAP_Pressure ] = R . set_pressure ;
} else if ( R . mode = = MODE_APAP ) {
if ( R . min_pressure > = 0 ) sess - > settings [ CPAP_PressureMin ] = R . min_pressure ;
if ( R . max_pressure > = 0 ) sess - > settings [ CPAP_PressureMax ] = R . max_pressure ;
} else if ( R . mode = = MODE_BILEVEL_FIXED ) {
if ( R . epap > = 0 ) sess - > settings [ CPAP_EPAP ] = R . epap ;
if ( R . ipap > = 0 ) sess - > settings [ CPAP_IPAP ] = R . ipap ;
if ( R . ps > = 0 ) sess - > settings [ CPAP_PS ] = R . ps ;
} else if ( R . mode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
if ( R . min_epap > = 0 ) sess - > settings [ CPAP_EPAPLo ] = R . min_epap ;
if ( R . max_ipap > = 0 ) sess - > settings [ CPAP_IPAPHi ] = R . max_ipap ;
if ( R . ps > = 0 ) sess - > settings [ CPAP_PS ] = R . ps ;
2022-11-12 19:46:57 +00:00
if ( R . s_Cycle > = 0 ) sess - > settings [ RMAS1x_Cycle ] = R . s_Cycle ;
if ( R . s_Trigger > = 0 ) sess - > settings [ RMAS1x_Trigger ] = R . s_Trigger ;
if ( R . s_TiMax > = 0 ) sess - > settings [ RMAS1x_TiMax ] = R . s_TiMax ;
if ( R . s_TiMin > = 0 ) sess - > settings [ RMAS1x_TiMin ] = R . s_TiMin ;
2019-12-24 02:44:10 +00:00
} else if ( R . mode = = MODE_ASV ) {
if ( R . epap > = 0 ) sess - > settings [ CPAP_EPAP ] = R . epap ;
if ( R . min_ps > = 0 ) sess - > settings [ CPAP_PSMin ] = R . min_ps ;
if ( R . max_ps > = 0 ) sess - > settings [ CPAP_PSMax ] = R . max_ps ;
if ( R . max_ipap > = 0 ) sess - > settings [ CPAP_IPAPHi ] = R . max_ipap ;
} else if ( R . mode = = MODE_ASV_VARIABLE_EPAP ) {
if ( R . max_epap > = 0 ) sess - > settings [ CPAP_EPAPHi ] = R . max_epap ;
if ( R . min_epap > = 0 ) sess - > settings [ CPAP_EPAPLo ] = R . min_epap ;
if ( R . max_ipap > = 0 ) sess - > settings [ CPAP_IPAPHi ] = R . max_ipap ;
if ( R . min_ipap > = 0 ) sess - > settings [ CPAP_IPAPLo ] = R . min_ipap ;
if ( R . min_ps > = 0 ) sess - > settings [ CPAP_PSMin ] = R . min_ps ;
if ( R . max_ps > = 0 ) sess - > settings [ CPAP_PSMax ] = R . max_ps ;
2021-09-22 17:52:02 +00:00
} else {
2022-01-02 01:49:40 +00:00
qDebug ( ) < < " Setting session pressures for R.mode " < < R . mode ;
2021-10-28 00:26:18 +00:00
if ( R . set_pressure > 0 ) sess - > settings [ CPAP_Pressure ] = R . set_pressure ;
if ( R . min_pressure > 0 ) sess - > settings [ CPAP_PressureMin ] = R . min_pressure ;
if ( R . max_pressure > 0 ) sess - > settings [ CPAP_PressureMax ] = R . max_pressure ;
if ( R . max_epap > 0 ) sess - > settings [ CPAP_EPAPHi ] = R . max_epap ;
if ( R . min_epap > 0 ) sess - > settings [ CPAP_EPAPLo ] = R . min_epap ;
if ( R . max_ipap > 0 ) sess - > settings [ CPAP_IPAPHi ] = R . max_ipap ;
if ( R . min_ipap > 0 ) sess - > settings [ CPAP_IPAPLo ] = R . min_ipap ;
if ( R . min_ps > 0 ) sess - > settings [ CPAP_PSMin ] = R . min_ps ;
if ( R . max_ps > 0 ) sess - > settings [ CPAP_PSMax ] = R . max_ps ;
if ( R . ps > 0 ) sess - > settings [ CPAP_PS ] = R . ps ;
if ( R . epap > 0 ) sess - > settings [ CPAP_EPAP ] = R . epap ;
if ( R . ipap > 0 ) sess - > settings [ CPAP_IPAP ] = R . ipap ;
2019-12-24 02:44:10 +00:00
}
}
2011-06-28 02:21:38 +00:00
2019-12-24 02:44:10 +00:00
if ( R . epr > = 0 ) {
sess - > settings [ RMS9_EPR ] = ( int ) R . epr ;
if ( R . epr > 0 ) {
if ( R . epr_level > = 0 ) {
sess - > settings [ RMS9_EPRLevel ] = ( int ) R . epr_level ;
}
}
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
if ( R . s_RampEnable > = 0 ) {
sess - > settings [ RMS9_RampEnable ] = R . s_RampEnable ;
2018-05-07 00:37:22 +00:00
2019-12-24 02:44:10 +00:00
if ( R . s_RampEnable > = 1 ) {
if ( R . s_RampTime > = 0 ) {
sess - > settings [ CPAP_RampTime ] = R . s_RampTime ;
}
2022-04-22 16:33:32 +00:00
if ( R . s_RampPressure > = 0 ) {
sess - > settings [ CPAP_RampPressure ] = R . s_RampPressure ;
2019-12-24 02:44:10 +00:00
}
}
}
2018-05-07 01:57:58 +00:00
2019-12-24 02:44:10 +00:00
if ( R . s_SmartStart > = 0 ) {
sess - > settings [ RMS9_SmartStart ] = R . s_SmartStart ;
}
2021-08-22 21:39:13 +00:00
if ( R . s_SmartStop > = 0 ) {
sess - > settings [ RMAS11_SmartStop ] = R . s_SmartStop ;
}
2019-12-24 02:44:10 +00:00
if ( R . s_ABFilter > = 0 ) {
sess - > settings [ RMS9_ABFilter ] = R . s_ABFilter ;
}
if ( R . s_ClimateControl > = 0 ) {
sess - > settings [ RMS9_ClimateControl ] = R . s_ClimateControl ;
}
if ( R . s_Mask > = 0 ) {
sess - > settings [ RMS9_Mask ] = R . s_Mask ;
}
if ( R . s_PtAccess > = 0 ) {
sess - > settings [ RMS9_PtAccess ] = R . s_PtAccess ;
}
2018-04-28 05:33:26 +00:00
2021-08-17 15:32:39 +00:00
if ( R . s_PtView > = 0 ) {
sess - > settings [ RMAS11_PtView ] = R . s_PtView ;
}
2019-12-24 02:44:10 +00:00
if ( R . s_HumEnable > = 0 ) {
sess - > settings [ RMS9_HumidStatus ] = ( short ) R . s_HumEnable ;
if ( ( R . s_HumEnable > = 1 ) & & ( R . s_HumLevel > = 0 ) ) {
sess - > settings [ RMS9_HumidLevel ] = ( short ) R . s_HumLevel ;
}
}
if ( R . s_TempEnable > = 0 ) {
sess - > settings [ RMS9_TempEnable ] = ( short ) R . s_TempEnable ;
if ( ( R . s_TempEnable > = 1 ) & & ( R . s_Temp > = 0 ) ) {
sess - > settings [ RMS9_Temp ] = ( short ) R . s_Temp ;
}
}
2021-07-09 21:39:23 +00:00
if ( R . s_Comfort > = 0 ) {
sess - > settings [ RMAS1x_Comfort ] = R . s_Comfort ;
}
2019-12-24 02:44:10 +00:00
}
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
struct OverlappingEDF {
quint32 start ;
quint32 end ;
QMultiMap < quint32 , QString > filemap ; // key is start time, value is filename
Session * sess ;
} ;
2018-05-07 00:37:22 +00:00
2019-12-24 02:44:10 +00:00
void ResDayTask : : run ( )
{
2020-08-08 18:17:54 +00:00
# ifdef SESSION_DEBUG
qDebug ( ) < < " Processing STR and edf files for " < < resday - > date ;
# endif
2019-12-24 02:44:10 +00:00
if ( resday - > files . size ( ) = = 0 ) { // No EDF files???
2020-05-07 20:43:52 +00:00
if ( ( ! resday - > str . date . isValid ( ) ) | | ( resday - > str . date > QDate : : currentDate ( ) ) ) {
2019-12-24 02:44:10 +00:00
// This condition should be impossible, but just in case something gets fudged up elsewhere later
qDebug ( ) < < " No edf files in resday " < < resday - > date < < " and the str date is inValid " ;
return ;
}
// Summary only day, create sessions for each mask-on/off pair and tag them summary only
STRRecord & R = resday - > str ;
2019-12-30 16:14:05 +00:00
# ifdef SESSION_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Creating summary-only sessions for " < < resday - > date ;
2019-12-30 16:14:05 +00:00
# endif
2019-12-24 02:44:10 +00:00
for ( int i = 0 ; i < resday - > str . maskon . size ( ) ; + + i ) {
quint32 maskon = resday - > str . maskon [ i ] ;
quint32 maskoff = resday - > str . maskoff [ i ] ;
2020-11-19 20:51:01 +00:00
/**
QTime noon ( 12 , 00 , 00 ) ;
QDateTime daybegin ( resday - > date , noon ) ; // Beginning of ResMed day
quint32 dayend = daybegin . addDays ( 1 ) . addMSecs ( - 1 ) . toTime_t ( ) ; // End of ResMed day
if ( ( maskon > dayend ) | |
( maskoff > dayend ) ) {
qWarning ( ) < < " mask time in future " < < resday - > date < < daybegin < < dayend < < " maskon " < < maskon < < " maskoff " < < maskoff ;
2020-05-07 20:43:52 +00:00
continue ;
}
2020-11-19 20:51:01 +00:00
* */
2019-12-24 02:44:10 +00:00
if ( ( ( maskon > 0 ) & & ( maskoff > 0 ) ) & & ( maskon ! = maskoff ) ) { //ignore very short sessions
Session * sess = new Session ( mach , maskon ) ;
sess - > set_first ( quint64 ( maskon ) * 1000L ) ;
sess - > set_last ( quint64 ( maskoff ) * 1000L ) ;
StoreSettings ( sess , R ) ; // Process the STR.edf settings
StoreSummaryStatistics ( sess , R ) ; // We want the summary information too
2018-04-28 05:33:26 +00:00
2019-12-24 02:44:10 +00:00
sess - > setSummaryOnly ( true ) ;
sess - > SetChanged ( true ) ;
2018-04-28 05:33:26 +00:00
2020-02-11 03:18:39 +00:00
// loader->sessionMutex.lock(); // This chunk moved into SaveSession below
// sess->Store(mach->getDataPath());
// mach->AddSession(sess);
// loader->sessionCount++;
// loader->sessionMutex.unlock();
//// delete sess;
save ( loader , sess ) ; // This is aliased to SaveSession - unless testing
2019-12-24 02:44:10 +00:00
}
}
2022-01-05 20:35:35 +00:00
// qDebug() << "Finished summary processing for" << resday->date;
2019-12-24 02:44:10 +00:00
return ;
2018-04-28 05:33:26 +00:00
}
2019-12-24 02:44:10 +00:00
// sooo... at this point we have
// resday record populated with correct STR.edf settings for this date
// files list containing unsorted EDF files that match this day
2022-02-27 16:50:10 +00:00
// guaranteed no sessions for this day for this device.
2018-05-29 08:22:01 +00:00
2019-12-30 16:14:05 +00:00
// Need to check overlapping files in session candidates
2014-07-28 13:56:29 +00:00
2019-12-24 02:44:10 +00:00
QList < OverlappingEDF > overlaps ;
2013-10-16 02:52:25 +00:00
2020-02-06 14:13:16 +00:00
int maskOnSize = resday - > str . maskon . size ( ) ;
2019-12-24 02:44:10 +00:00
if ( resday - > str . date . isValid ( ) ) {
//First populate Overlaps with Mask ON/OFF events
2020-02-06 14:13:16 +00:00
for ( int i = 0 ; i < maskOnSize ; + + i ) {
2020-11-19 20:51:01 +00:00
// if ( (resday->str.maskon[i] > QDateTime::currentDateTime().toTime_t()) ||
// (resday->str.maskoff[i] > QDateTime::currentDateTime().toTime_t()) ) {
// qWarning() << "mask time in future" << resday->date << "now" << QDateTime::currentDateTime().toTime_t() << "maskon" << resday->str.maskon[i] << "maskoff" << resday->str.maskoff[i];
// continue;
// }
/*
QTime noon ( 12 , 00 , 00 ) ;
QDateTime daybegin ( resday - > date , noon ) ; // Beginning of ResMed day
quint32 dayend = daybegin . addDays ( 1 ) . addMSecs ( - 1 ) . toTime_t ( ) ; // End of ResMed day
if ( ( resday - > str . maskon [ i ] > dayend ) | |
( resday - > str . maskoff [ i ] > dayend ) ) {
qWarning ( ) < < " mask time in future " < < resday - > date < < " daybegin: " < < daybegin < < " dayend: " < < dayend < < " maskon " < < resday - > str . maskon [ i ] < < " maskoff " < < resday - > str . maskoff [ i ] ;
2020-05-07 20:43:52 +00:00
continue ;
}
2020-11-19 20:51:01 +00:00
*/
if ( ( ( resday - > str . maskon [ i ] > 0 ) | | ( resday - > str . maskoff [ i ] > 0 ) )
2020-02-10 04:45:31 +00:00
& & ( resday - > str . maskon [ i ] ! = resday - > str . maskoff [ i ] ) ) {
2019-12-24 02:44:10 +00:00
OverlappingEDF ov ;
ov . start = resday - > str . maskon [ i ] ;
ov . end = resday - > str . maskoff [ i ] ;
ov . sess = nullptr ;
overlaps . append ( ov ) ;
}
}
}
2020-08-20 00:12:41 +00:00
# ifdef STR_DEBUG
2020-08-19 17:14:02 +00:00
if ( overlaps . size ( ) > 0 )
qDebug ( ) . noquote ( ) < < " Created " < < overlaps . size ( ) < < " sessionGroups from STR record for " < < resday - > str . date . toString ( ) ;
2020-08-20 00:12:41 +00:00
# endif
2013-10-16 02:52:25 +00:00
2019-12-24 02:44:10 +00:00
QMap < quint32 , QString > EVElist , CSLlist ;
for ( auto f_itr = resday - > files . begin ( ) , fend = resday - > files . end ( ) ; f_itr ! = fend ; + + f_itr ) {
const QString & filename = f_itr . key ( ) ;
const QString & fullpath = f_itr . value ( ) ;
// QString ext = filename.section("_", -1).section(".",0,0).toUpper();
EDFType type = lookupEDFType ( filename ) ;
2014-07-30 20:25:06 +00:00
2019-12-24 02:44:10 +00:00
QString datestr = filename . section ( " _ " , 0 , 1 ) ;
2020-02-10 04:45:31 +00:00
// QDateTime filetime = QDateTime().fromString(datestr,"yyyyMMdd_HHmmss");
QDate d2 = QDate : : fromString ( datestr . left ( 8 ) , " yyyyMMdd " ) ;
QTime t2 = QTime : : fromString ( datestr . right ( 6 ) , " hhmmss " ) ;
QDateTime filetime = QDateTime ( d2 , t2 , EDFInfo : : localNoDST ) ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
quint32 filetime_t = filetime . toTime_t ( ) ;
2019-12-30 16:14:05 +00:00
if ( type = = EDF_EVE ) { // skip the EVE and CSL files, b/c they often cover all sessions
2019-12-24 02:44:10 +00:00
EVElist [ filetime_t ] = filename ;
continue ;
} else if ( type = = EDF_CSL ) {
CSLlist [ filetime_t ] = filename ;
continue ;
}
bool added = false ;
for ( auto & ovr : overlaps ) {
if ( ( filetime_t > = ( ovr . start ) ) & & ( filetime_t < ovr . end ) ) {
ovr . filemap . insert ( filetime_t , filename ) ;
added = true ;
break ;
}
}
2020-08-08 18:17:54 +00:00
if ( ! added ) { // Didn't get a hit, look at the EDF files duration and check for an overlap
2019-12-24 02:44:10 +00:00
EDFduration dur = getEDFDuration ( fullpath ) ;
2020-11-19 20:51:01 +00:00
/**
QTime noon ( 12 , 00 , 00 ) ;
QDateTime daybegin ( resday - > date , noon ) ; // Beginning of ResMed day
quint32 dayend = daybegin . addDays ( 1 ) . addMSecs ( - 1 ) . toTime_t ( ) ; // End of ResMed day
if ( ( dur . start > ( dayend ) ) | |
( dur . end > ( dayend ) ) ) {
qWarning ( ) < < " Future Date in " < < fullpath < < " dayend " < < dayend < < " dur.start " < < dur . start < < " dur.end " < < dur . end ;
continue ; // skip this file
}
* */
2019-12-24 02:44:10 +00:00
for ( int i = overlaps . size ( ) - 1 ; i > = 0 ; - - i ) {
OverlappingEDF & ovr = overlaps [ i ] ;
if ( ( ovr . start < dur . end ) & & ( dur . start < ovr . end ) ) {
ovr . filemap . insert ( filetime_t , filename ) ;
added = true ;
2019-12-30 16:14:05 +00:00
# ifdef SESSION_DEBUG
qDebug ( ) < < " Adding " < < filename < < " to overlap " < < i ;
qDebug ( ) < < " Overlap starts: " < < ovr . start < < " ends: " < < ovr . end ;
qDebug ( ) < < " File time starts: " < < dur . start < < " ends: " < < dur . end ;
# endif
// Expand ovr's scope -- I think this is necessary!! (PO)
// YES! when the STR file is missing, there are no mask on/off entries
// and the edf files are not always created at the same time
ovr . start = min ( ovr . start , dur . start ) ;
ovr . end = max ( ovr . end , dur . end ) ;
// if ( (dur.start < ovr.start) || (dur.end > ovr.end) )
// qDebug() << "Should have expanded overlap" << i << "for" << filename;
2019-12-24 02:44:10 +00:00
break ;
}
2019-12-30 16:14:05 +00:00
} // end for walk existing overlap entries
2019-12-24 02:44:10 +00:00
if ( ! added ) {
2019-12-30 16:14:05 +00:00
if ( dur . start ! = dur . end ) { // Didn't fit it in anywhere, create a new Overlap entry/session
2020-02-02 19:29:23 +00:00
OverlappingEDF ov ;
ov . start = dur . start ;
ov . end = dur . end ;
ov . filemap . insert ( filetime_t , filename ) ;
2019-12-30 16:14:05 +00:00
# ifdef SESSION_DEBUG
2020-08-08 18:17:54 +00:00
qDebug ( ) < < " Creating overlap for " < < filename < < " missing STR record " ;
2020-02-02 19:29:23 +00:00
qDebug ( ) < < " Starts: " < < dur . start < < " Ends: " < < dur . end ;
2019-12-30 16:14:05 +00:00
# endif
overlaps . append ( ov ) ;
2020-02-02 19:29:23 +00:00
} else {
2019-12-30 16:14:05 +00:00
# ifdef SESSION_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Skipping zero duration file " < < filename ;
2019-12-30 16:14:05 +00:00
# endif
2019-12-24 02:44:10 +00:00
}
2019-12-30 16:14:05 +00:00
} // end create a new overlap entry
} // end check for file overlap
} // end for walk resday files list
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
// Create an ordered map and see how far apart the sessions really are.
QMap < quint32 , OverlappingEDF > mapov ;
for ( auto & ovr : overlaps ) {
mapov [ ovr . start ] = ovr ;
}
2018-05-06 16:59:50 +00:00
2019-12-30 16:14:05 +00:00
// We are not going to merge close sessions - gaps can be useful markers for users
// // Examine the gaps in between to see if we should merge sessions
// for (auto oit=mapov.begin(), oend=mapov.end(); oit != oend; ++oit) {
// // Get next in line
// auto next_oit = oit+1;
// if (next_oit != mapov.end()) {
// OverlappingEDF & A = oit.value();
// OverlappingEDF & B = next_oit.value();
// int gap = B.start - A.end;
// if (gap < 60) { // TODO see if we should use the prefs value here... ???
// // qDebug() << "Only a" << gap << "s sgap between ResMed sessions on" << resday->date.toString();
// }
// }
// }
2011-12-21 04:25:01 +00:00
2020-08-08 18:17:54 +00:00
if ( overlaps . size ( ) = = 0 ) {
qDebug ( ) < < " No sessionGroups for " < < resday - > date < < " FINSIHED " ;
2019-12-24 02:44:10 +00:00
return ;
2020-08-08 18:17:54 +00:00
}
2014-05-20 11:51:47 +00:00
2019-12-24 02:44:10 +00:00
// Now overlaps is populated with zero or more individual session groups of EDF files (zero because of sucky summary only days)
for ( auto & ovr : overlaps ) {
if ( ovr . filemap . size ( ) = = 0 )
continue ;
Session * sess = new Session ( mach , ovr . start ) ;
2020-03-24 14:34:50 +00:00
// Do not set the session times according to Mask on/off times
// The LoadXXX edf routines will update them with recording start and durations
// sess->set_first(quint64(ovr.start)*1000L);
// sess->set_last(quint64(ovr.end)*1000L);
2019-12-24 02:44:10 +00:00
ovr . sess = sess ;
2014-05-20 11:51:47 +00:00
2019-12-24 02:44:10 +00:00
for ( auto mit = ovr . filemap . begin ( ) , mend = ovr . filemap . end ( ) ; mit ! = mend ; + + mit ) {
const QString & filename = mit . value ( ) ;
const QString & fullpath = resday - > files [ filename ] ;
EDFType type = lookupEDFType ( filename ) ;
2019-07-22 21:01:47 +00:00
2019-12-24 02:44:10 +00:00
# ifdef SESSION_DEBUG
sess - > session_files . append ( filename ) ;
# endif
switch ( type ) {
case EDF_BRP :
loader - > LoadBRP ( sess , fullpath ) ;
break ;
case EDF_PLD :
loader - > LoadPLD ( sess , fullpath ) ;
break ;
case EDF_SAD :
2023-10-13 23:57:09 +00:00
case EDF_SA2 :
2019-12-24 02:44:10 +00:00
loader - > LoadSAD ( sess , fullpath ) ;
break ;
case EDF_EVE :
case EDF_CSL :
2020-03-23 18:41:31 +00:00
case EDF_AEV : // this is in the 36039 - must figure out what to do with it
2019-12-24 02:44:10 +00:00
break ;
default :
qWarning ( ) < < " Unrecognized file type for " < < filename ;
}
2020-08-08 18:17:54 +00:00
} // end for each edf file in the sessionGroup
2019-12-24 02:44:10 +00:00
// Turns out there is only one or sometimes two EVE's per day, and they store data for the whole day
// So we have to extract Annotations data and apply it for all sessions
for ( auto eit = EVElist . begin ( ) , eveend = EVElist . end ( ) ; eit ! = eveend ; + + eit ) {
const QString & fullpath = resday - > files [ eit . value ( ) ] ;
loader - > LoadEVE ( ovr . sess , fullpath ) ;
}
for ( auto eit = CSLlist . begin ( ) , cslend = CSLlist . end ( ) ; eit ! = cslend ; + + eit ) {
const QString & fullpath = resday - > files [ eit . value ( ) ] ;
loader - > LoadCSL ( ovr . sess , fullpath ) ;
}
2011-06-29 17:58:28 +00:00
2019-12-24 02:44:10 +00:00
if ( EVElist . size ( ) = = 0 ) {
sess - > AddEventList ( CPAP_Obstructive , EVL_Event ) ;
sess - > AddEventList ( CPAP_ClearAirway , EVL_Event ) ;
sess - > AddEventList ( CPAP_Apnea , EVL_Event ) ;
sess - > AddEventList ( CPAP_Hypopnea , EVL_Event ) ;
}
sess - > setSummaryOnly ( false ) ;
sess - > SetChanged ( true ) ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
if ( sess - > length ( ) = = 0 ) {
// we want empty sessions even though they are crap
qDebug ( ) < < " Session " < < sess - > session ( )
2020-02-10 04:45:31 +00:00
< < " [ " + QDateTime : : fromTime_t ( sess - > session ( ) ) . toString ( " MMM dd, yyyy hh:mm:ss " ) + " ] "
2020-08-19 17:14:02 +00:00
< < " has zero duration " < < QString ( " Start: %1 " ) . arg ( sess - > realFirst ( ) , 0 , 16 ) < < QString ( " End: %1 " ) . arg ( sess - > realLast ( ) , 0 , 16 ) ;
2020-02-10 04:45:31 +00:00
}
if ( sess - > length ( ) < 0 ) {
// we want empty sessions even though they are crap
qDebug ( ) < < " Session " < < sess - > session ( )
< < " [ " + QDateTime : : fromTime_t ( sess - > session ( ) ) . toString ( " MMM dd, yyyy hh:mm:ss " ) + " ] "
< < " has negative duration " ;
2020-05-07 20:43:52 +00:00
qDebug ( ) < < QString ( " Start: %1 " ) . arg ( sess - > realFirst ( ) , 0 , 16 ) < < QString ( " End: %1 " ) . arg ( sess - > realLast ( ) , 0 , 16 ) ;
2019-12-24 02:44:10 +00:00
}
2018-05-03 09:59:31 +00:00
2019-12-24 02:44:10 +00:00
if ( resday - > str . date . isValid ( ) ) {
STRRecord & R = resday - > str ;
2014-07-27 16:35:49 +00:00
2019-12-24 02:44:10 +00:00
// Claim this session
R . sessionid = sess - > session ( ) ;
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
// Save maskon time in session setting so we can use it later to avoid doubleups.
//sess->settings[RMS9_MaskOnTime] = R.maskon;
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
# ifdef SESSION_DEBUG
sess - > session_files . append ( " STR.edf " ) ;
# endif
StoreSettings ( sess , R ) ;
2013-10-22 11:42:57 +00:00
2020-05-07 20:43:52 +00:00
} else { // No corresponding STR.edf record, but we have EDF files
# ifdef STR_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " EDF files without STR record " < < resday - > date . toString ( ) ;
2020-05-07 20:43:52 +00:00
# endif
2019-12-24 02:44:10 +00:00
bool foundprev = false ;
loader - > sessionMutex . lock ( ) ;
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
auto it = p_profile - > daylist . find ( resday - > date ) ; // should exist already to be here
auto begin = p_profile - > daylist . begin ( ) ;
while ( it ! = begin ) {
- - it ;
Day * day = it . value ( ) ;
bool hasmachine = day & & day - > hasMachine ( mach ) ;
2018-05-03 09:59:31 +00:00
2020-08-08 18:17:54 +00:00
if ( ! hasmachine )
2019-12-24 02:44:10 +00:00
continue ;
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
QList < Session * > sessions = day - > getSessions ( MT_CPAP ) ;
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
if ( sessions . size ( ) > 0 ) {
Session * chksess = sessions [ 0 ] ;
sess - > settings = chksess - > settings ;
foundprev = true ;
break ;
}
}
loader - > sessionMutex . unlock ( ) ;
sess - > setNoSettings ( true ) ;
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
if ( ! foundprev ) {
// We have no Summary or Settings data... we need to do something to indicate this, and detect the mode
if ( sess - > channelDataExists ( CPAP_Pressure ) ) {
qDebug ( ) < < " Guessing the PAP mode... " ;
GuessPAPMode ( sess ) ;
}
}
2020-05-07 20:43:52 +00:00
} // end else no STR record for these edf files
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
sess - > UpdateSummaries ( ) ;
2019-12-30 16:14:05 +00:00
# ifdef SESSION_DEBUG
2019-12-24 02:44:10 +00:00
qDebug ( ) < < " Adding session " < < sess - > session ( )
2020-05-07 20:43:52 +00:00
< < " [ " + QDateTime : : fromTime_t ( sess - > session ( ) ) . toString ( " MMM dd, yyyy hh:mm:ss " ) + " ] " ;
2019-12-30 16:14:05 +00:00
# endif
2013-10-22 11:42:57 +00:00
2019-12-24 02:44:10 +00:00
// Save is not threadsafe? (meh... it seems to be)
// loader->saveMutex.lock();
// loader->saveMutex.unlock();
2013-10-22 11:42:57 +00:00
2020-11-19 20:51:01 +00:00
// if ( (QDateTime::fromTime_t(sess->session()) > QDateTime::currentDateTime()) ||
if ( ( sess - > realFirst ( ) = = 0 ) | | ( sess - > realLast ( ) = = 0 ) )
2020-08-20 00:12:41 +00:00
qWarning ( ) . noquote ( ) < < " Skipping future or absent date session: " < < sess - > session ( )
2020-05-07 20:43:52 +00:00
< < " [ " + QDateTime : : fromTime_t ( sess - > session ( ) ) . toString ( " MMM dd, yyyy hh:mm:ss " ) + " ] "
2020-08-19 17:14:02 +00:00
< < " \n original date is " < < resday - > date . toString ( )
< < " session realFirst " < < sess - > realFirst ( ) < < " realLast " < < sess - > realLast ( ) ;
2020-05-07 20:43:52 +00:00
else
save ( loader , sess ) ;
2019-12-24 02:44:10 +00:00
// Free the memory used by this session
sess - > TrashEvents ( ) ;
2020-02-10 04:45:31 +00:00
// delete sess;
2019-12-24 02:44:10 +00:00
} // end for-loop walking the overlaps (file groups per session
2013-10-22 11:42:57 +00:00
}
2020-02-10 21:04:03 +00:00
void ResmedLoader : : SaveSession ( ResmedLoader * loader , Session * sess )
{
Machine * mach = sess - > machine ( ) ;
2020-08-08 18:17:54 +00:00
loader - > sessionMutex . lock ( ) ; // AddSession definitely ain't threadsafe.
if ( ! sess - > Store ( mach - > getDataPath ( ) ) ) {
qWarning ( ) < < " Failed to store session " < < sess - > session ( ) ;
}
if ( ! mach - > AddSession ( sess ) ) {
qWarning ( ) < < " Session " < < sess - > session ( ) < < " was not addded " ;
}
2020-02-10 21:04:03 +00:00
loader - > sessionCount + + ;
loader - > sessionMutex . unlock ( ) ;
}
2019-12-24 02:44:10 +00:00
bool matchSignal ( ChannelID ch , const QString & name ) ; // forward
2014-09-22 04:32:15 +00:00
bool ResmedLoader : : LoadCSL ( Session * sess , const QString & path )
{
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
QTime time ;
time . start ( ) ;
# endif
2019-07-22 21:01:47 +00:00
2020-05-07 20:43:52 +00:00
QString filename = path . section ( - 2 , - 1 ) ;
2019-07-31 18:36:40 +00:00
ResMedEDFInfo edf ;
2020-01-31 00:54:53 +00:00
if ( ! edf . Open ( path ) ) {
2020-05-07 20:43:52 +00:00
qDebug ( ) < < " LoadCSL failed to open " < < filename ;
2020-02-02 19:29:23 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2019-07-22 21:01:47 +00:00
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfopentime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2019-07-22 21:01:47 +00:00
2020-01-31 00:54:53 +00:00
if ( ! edf . Parse ( ) ) {
2020-05-07 20:43:52 +00:00
qDebug ( ) < < " LoadCSL failed to parse " < < filename ;
2014-09-22 04:32:15 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2019-07-22 21:01:47 +00:00
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfparsetime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2014-09-22 04:32:15 +00:00
2022-02-18 01:59:16 +00:00
// Always create CSR event list so that overview always finds something
EventList * CSR = sess - > AddEventList ( CPAP_CSR , EVL_Event ) ;
2014-09-22 04:32:15 +00:00
// Allow for empty sessions..
qint64 csr_starts = 0 ;
// Process event annotation records
2019-09-25 11:25:44 +00:00
// qDebug() << "File has " << edf.annotations.size() << "annotation vectors";
2019-08-30 17:22:19 +00:00
// int vec = 1;
2019-08-16 00:30:54 +00:00
for ( auto annoVec = edf . annotations . begin ( ) ; annoVec ! = edf . annotations . end ( ) ; annoVec + + ) {
2019-08-30 17:22:19 +00:00
// qDebug() << "Vector " << vec++ << " has " << annoVec->size() << " annotations";
2019-08-16 00:30:54 +00:00
for ( auto anno = annoVec - > begin ( ) ; anno ! = annoVec - > end ( ) ; anno + + ) {
2019-08-30 17:22:19 +00:00
// qDebug() << "Offset: " << anno->offset << " Duration: " << anno->duration << " Text: " << anno->text;
2020-08-09 21:17:21 +00:00
qint64 tt = edf . startdate + qint64 ( anno - > offset * 1000L ) ;
2019-08-16 00:30:54 +00:00
if ( ! anno - > text . isEmpty ( ) ) {
2019-09-25 11:25:44 +00:00
if ( anno - > text = = " CSR Start " ) {
2019-08-16 00:30:54 +00:00
csr_starts = tt ;
2019-09-25 11:25:44 +00:00
} else if ( anno - > text = = " CSR End " ) {
2022-02-18 01:59:16 +00:00
// if ( ! CSR) {
// CSR = sess->AddEventList(CPAP_CSR, EVL_Event);
// }
2019-08-16 00:30:54 +00:00
if ( csr_starts > 0 ) {
if ( sess - > checkInside ( csr_starts ) ) {
CSR - > AddEvent ( tt , double ( tt - csr_starts ) / 1000.0 ) ;
2014-09-22 04:32:15 +00:00
}
2019-08-16 00:30:54 +00:00
csr_starts = 0 ;
} else {
2020-08-09 21:17:21 +00:00
qWarning ( ) < < " Split csr event flag in " < < edf . filename ;
2014-09-22 04:32:15 +00:00
}
2019-08-30 17:22:19 +00:00
} else if ( anno - > text ! = " Recording starts " ) {
2020-08-09 21:17:21 +00:00
qWarning ( ) < < " Unobserved ResMed CSL annotation field: " < < anno - > text ;
2014-09-22 04:32:15 +00:00
}
}
}
2019-08-16 00:30:54 +00:00
}
2014-09-22 04:32:15 +00:00
2019-08-16 00:30:54 +00:00
if ( csr_starts > 0 ) {
qDebug ( ) < < " Unfinished csr event in " < < edf . filename ;
2014-09-22 04:32:15 +00:00
}
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
timeMutex . lock ( ) ;
timeInLoadCSL + = time . elapsed ( ) ;
timeInEDFOpen + = edfopentime ;
2019-07-31 18:36:40 +00:00
timeInEDFInfo + = edfparsetime ;
2018-05-06 16:59:50 +00:00
timeMutex . unlock ( ) ;
# endif
2014-10-08 16:51:09 +00:00
2014-09-22 04:32:15 +00:00
return true ;
}
2013-10-22 11:42:57 +00:00
2014-05-20 11:51:47 +00:00
bool ResmedLoader : : LoadEVE ( Session * sess , const QString & path )
2011-06-29 14:19:38 +00:00
{
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
QTime time ;
time . start ( ) ;
# endif
2020-05-07 20:43:52 +00:00
QString filename = path . section ( - 2 , - 1 ) ;
2019-07-31 18:36:40 +00:00
ResMedEDFInfo edf ;
2020-01-31 00:54:53 +00:00
if ( ! edf . Open ( path ) ) {
2020-05-07 20:43:52 +00:00
qDebug ( ) < < " LoadEVE failed to open " < < filename ;
2020-02-02 19:29:23 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfopentime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2020-01-31 00:54:53 +00:00
if ( ! edf . Parse ( ) ) {
2020-05-07 20:43:52 +00:00
qDebug ( ) < < " LoadEVE failed to parse " < < filename ;
2014-05-20 11:51:47 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfparsetime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2011-07-27 09:21:53 +00:00
2014-05-04 18:02:41 +00:00
// Notes: Event records have useless duration record.
2018-04-28 05:33:26 +00:00
// Do not update session start / end times because they are needed to determine if events belong in this session or not...
2012-01-12 05:25:51 +00:00
2014-09-17 17:20:01 +00:00
EventList * OA = nullptr , * HY = nullptr , * CA = nullptr , * UA = nullptr , * RE = nullptr ;
2012-01-12 05:25:51 +00:00
// Allow for empty sessions..
2014-05-04 18:02:41 +00:00
2019-08-16 00:30:54 +00:00
// Create some EventLists
2014-04-17 05:58:57 +00:00
OA = sess - > AddEventList ( CPAP_Obstructive , EVL_Event ) ;
HY = sess - > AddEventList ( CPAP_Hypopnea , EVL_Event ) ;
UA = sess - > AddEventList ( CPAP_Apnea , EVL_Event ) ;
2012-01-12 05:25:51 +00:00
2014-05-04 18:02:41 +00:00
// Process event annotation records
2019-09-25 11:25:44 +00:00
// qDebug() << "File has " << edf.annotations.size() << "annotation vectors";
2019-08-30 17:22:19 +00:00
// int vec = 1;
2019-08-16 00:30:54 +00:00
for ( auto annoVec = edf . annotations . begin ( ) ; annoVec ! = edf . annotations . end ( ) ; annoVec + + ) {
2019-08-30 17:22:19 +00:00
// qDebug() << "Vector " << vec++ << " has " << annoVec->size() << " annotations";
2019-08-16 00:30:54 +00:00
for ( auto anno = annoVec - > begin ( ) ; anno ! = annoVec - > end ( ) ; anno + + ) {
2020-08-09 21:17:21 +00:00
qint64 tt = edf . startdate + qint64 ( anno - > offset * 1000L ) ;
2019-08-30 17:22:19 +00:00
// qDebug() << "Offset: " << anno->offset << " Duration: " << anno->duration << " Text: " << anno->text;
2020-08-09 21:17:21 +00:00
// qDebug() << "Time: " << (tt/1000L). << " Duration: " << anno->duration << " Text: " << anno->text;
2019-08-16 00:30:54 +00:00
if ( ! anno - > text . isEmpty ( ) ) {
if ( matchSignal ( CPAP_Obstructive , anno - > text ) ) {
if ( sess - > checkInside ( tt ) )
OA - > AddEvent ( tt , anno - > duration ) ;
} else if ( matchSignal ( CPAP_Hypopnea , anno - > text ) ) {
if ( sess - > checkInside ( tt ) )
2019-12-24 02:44:10 +00:00
HY - > AddEvent ( tt , anno - > duration ) ; // Hyponeas may not have any duration!
2019-08-16 00:30:54 +00:00
} else if ( matchSignal ( CPAP_Apnea , anno - > text ) ) {
if ( sess - > checkInside ( tt ) )
UA - > AddEvent ( tt , anno - > duration ) ;
} else if ( matchSignal ( CPAP_RERA , anno - > text ) ) {
2022-02-27 16:50:10 +00:00
// Not all devices have it, so only create it when necessary..
2020-08-09 21:17:21 +00:00
if ( ! RE )
RE = sess - > AddEventList ( CPAP_RERA , EVL_Event ) ;
2019-08-16 00:30:54 +00:00
if ( sess - > checkInside ( tt ) )
RE - > AddEvent ( tt , anno - > duration ) ;
} else if ( matchSignal ( CPAP_ClearAirway , anno - > text ) ) {
2022-02-27 16:50:10 +00:00
// Not all devices have it, so only create it when necessary..
2020-08-09 21:17:21 +00:00
if ( ! CA )
CA = sess - > AddEventList ( CPAP_ClearAirway , EVL_Event ) ;
2019-08-16 00:30:54 +00:00
if ( sess - > checkInside ( tt ) )
CA - > AddEvent ( tt , anno - > duration ) ;
2021-09-12 01:33:01 +00:00
} else if ( anno - > text = = " SpO2 Desaturation " ) { // Used in 28509
continue ; // ignored for now
2019-08-16 00:30:54 +00:00
} else {
2019-08-30 17:22:19 +00:00
if ( anno - > text ! = " Recording starts " ) {
2019-08-16 00:30:54 +00:00
qDebug ( ) < < " Unobserved ResMed annotation field: " < < anno - > text ;
2011-06-29 17:58:28 +00:00
}
}
}
2014-04-17 05:58:57 +00:00
2011-06-29 17:58:28 +00:00
}
2011-06-29 14:19:38 +00:00
}
2019-08-16 00:30:54 +00:00
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
timeMutex . lock ( ) ;
timeInLoadEVE + = time . elapsed ( ) ;
timeInEDFOpen + = edfopentime ;
2019-07-31 18:36:40 +00:00
timeInEDFInfo + = edfparsetime ;
2018-05-06 16:59:50 +00:00
timeMutex . unlock ( ) ;
# endif
2014-04-17 05:58:57 +00:00
2011-07-01 10:10:44 +00:00
return true ;
2011-06-29 14:19:38 +00:00
}
2014-05-20 11:51:47 +00:00
bool ResmedLoader : : LoadBRP ( Session * sess , const QString & path )
2011-06-29 14:19:38 +00:00
{
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
QTime time ;
time . start ( ) ;
# endif
2020-05-07 20:43:52 +00:00
QString filename = path . section ( - 2 , - 1 ) ;
2019-07-31 18:36:40 +00:00
ResMedEDFInfo edf ;
2020-01-31 00:54:53 +00:00
if ( ! edf . Open ( path ) ) {
2020-05-15 23:58:11 +00:00
qDebug ( ) < < " LoadBRP failed to open " < < filename . section ( " / " , - 2 , - 1 ) ;
2020-02-02 19:29:23 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfopentime = time . elapsed ( ) ;
time . start ( ) ;
2019-12-24 02:44:10 +00:00
# endif
2020-01-31 00:54:53 +00:00
if ( ! edf . Parse ( ) ) {
2020-05-15 23:58:11 +00:00
# ifdef EDF_DEBUG
qDebug ( ) < < " LoadBRP failed to parse " < < filename . section ( " / " , - 2 , - 1 ) ;
# endif
2019-12-24 02:44:10 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2019-12-24 02:44:10 +00:00
# ifdef DEBUG_EFFICIENCY
int edfparsetime = time . elapsed ( ) ;
time . start ( ) ;
int AddWavetime = 0 ;
2022-01-10 23:02:25 +00:00
QTime time2 ;
2019-12-24 02:44:10 +00:00
# endif
sess - > updateFirst ( edf . startdate ) ;
2013-10-25 10:39:30 +00:00
2019-12-24 02:44:10 +00:00
qint64 duration = edf . GetNumDataRecords ( ) * edf . GetDurationMillis ( ) ;
sess - > updateLast ( edf . startdate + duration ) ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
for ( auto & es : edf . edfsignals ) {
long recs = es . sampleCnt * edf . GetNumDataRecords ( ) ;
if ( recs < 0 )
continue ;
ChannelID code ;
2012-01-09 15:38:41 +00:00
2019-12-24 02:44:10 +00:00
if ( matchSignal ( CPAP_FlowRate , es . label ) ) {
code = CPAP_FlowRate ;
es . gain * = 60.0 ;
es . physical_minimum * = 60.0 ;
es . physical_maximum * = 60.0 ;
es . physical_dimension = " L/M " ;
2011-06-29 20:30:23 +00:00
2019-12-24 02:44:10 +00:00
} else if ( matchSignal ( CPAP_MaskPressureHi , es . label ) ) {
code = CPAP_MaskPressureHi ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
} else if ( matchSignal ( CPAP_RespEvent , es . label ) ) {
code = CPAP_RespEvent ;
2011-06-29 20:30:23 +00:00
2020-02-22 02:41:00 +00:00
// } else if (es.label == "TrigCycEvt.40ms") { // we need a real code for this signal
// code = CPAP_TriggerEvent; // Well, it got folded into RespEvent
// continue;
2019-12-24 02:44:10 +00:00
} else if ( es . label ! = " Crc16 " ) {
qDebug ( ) < < " Unobserved ResMed BRP Signal " < < es . label ;
continue ;
} else
continue ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
if ( code ) {
double rate = double ( duration ) / double ( recs ) ;
EventList * a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
a - > setDimension ( es . physical_dimension ) ;
# ifdef DEBUG_EFFICIENCY
time2 . start ( ) ;
# endif
a - > AddWaveform ( edf . startdate , es . dataArray , recs , duration ) ;
# ifdef DEBUG_EFFICIENCY
AddWavetime + = time2 . elapsed ( ) ;
# endif
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
EventDataType min = a - > Min ( ) ;
EventDataType max = a - > Max ( ) ;
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
// Cap to physical dimensions, because there can be ram glitches/whatever that throw really big outliers.
if ( min < es . physical_minimum )
min = es . physical_minimum ;
if ( max > es . physical_maximum )
max = es . physical_maximum ;
2014-04-17 05:58:57 +00:00
2020-05-15 01:10:03 +00:00
sess - > updateMin ( code , min ) ;
sess - > updateMax ( code , max ) ;
2019-12-24 02:44:10 +00:00
sess - > setPhysMin ( code , es . physical_minimum ) ;
sess - > setPhysMax ( code , es . physical_maximum ) ;
}
2011-06-29 20:30:23 +00:00
}
2012-01-05 06:54:07 +00:00
# ifdef DEBUG_EFFICIENCY
2018-05-06 16:59:50 +00:00
timeMutex . lock ( ) ;
2019-12-24 02:44:10 +00:00
timeInLoadBRP + = time . elapsed ( ) ;
timeInEDFOpen + = edfopentime ;
timeInEDFInfo + = edfparsetime ;
timeInAddWaveform + = AddWavetime ;
2018-05-06 16:59:50 +00:00
timeMutex . unlock ( ) ;
2012-01-05 06:54:07 +00:00
# endif
2019-12-24 02:44:10 +00:00
return true ;
}
2014-05-04 18:02:41 +00:00
// Load SAD Oximetry Signals
2014-05-20 11:51:47 +00:00
bool ResmedLoader : : LoadSAD ( Session * sess , const QString & path )
2011-06-29 14:19:38 +00:00
{
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
QTime time ;
time . start ( ) ;
# endif
2019-07-22 21:01:47 +00:00
2020-05-07 20:43:52 +00:00
QString filename = path . section ( - 2 , - 1 ) ;
2019-07-31 18:36:40 +00:00
ResMedEDFInfo edf ;
2020-01-31 00:54:53 +00:00
if ( ! edf . Open ( path ) ) {
2020-05-15 23:58:11 +00:00
qDebug ( ) < < " LoadSAD failed to open " < < filename . section ( " / " , - 2 , - 1 ) ;
2020-02-02 19:29:23 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2019-07-22 21:01:47 +00:00
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfopentime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2019-07-22 21:01:47 +00:00
2020-01-31 00:54:53 +00:00
if ( ! edf . Parse ( ) ) {
2020-05-15 23:58:11 +00:00
# ifdef EDF_DEBUG
qDebug ( ) < < " LoadSAD failed to parse " < < filename . section ( " / " , - 2 , - 1 ) ;
# endif
2014-05-20 11:51:47 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2019-07-22 21:01:47 +00:00
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfparsetime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2014-05-20 11:51:47 +00:00
2011-08-06 13:37:06 +00:00
sess - > updateFirst ( edf . startdate ) ;
2019-07-31 18:36:40 +00:00
qint64 duration = edf . GetNumDataRecords ( ) * edf . GetDurationMillis ( ) ;
2014-04-17 05:58:57 +00:00
sess - > updateLast ( edf . startdate + duration ) ;
2011-08-06 13:37:06 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & es : edf . edfsignals ) {
2011-08-07 01:26:28 +00:00
//qDebug() << "SAD:" << es.label << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum;
2019-08-03 14:59:08 +00:00
long recs = es . sampleCnt * edf . GetNumDataRecords ( ) ;
2011-08-06 13:37:06 +00:00
ChannelID code ;
2014-04-17 05:58:57 +00:00
bool hasdata = false ;
2014-05-04 18:02:41 +00:00
for ( int i = 0 ; i < recs ; + + i ) {
2019-08-03 14:59:08 +00:00
if ( es . dataArray [ i ] ! = - 1 ) {
2014-04-17 05:58:57 +00:00
hasdata = true ;
2011-08-06 13:37:06 +00:00
break ;
}
}
2019-07-22 21:01:47 +00:00
if ( ! hasdata )
2014-05-20 11:51:47 +00:00
continue ;
2014-04-17 05:58:57 +00:00
2014-05-13 02:56:51 +00:00
if ( matchSignal ( OXI_Pulse , es . label ) ) {
code = OXI_Pulse ;
2014-05-04 18:02:41 +00:00
ToTimeDelta ( sess , edf , es , code , recs , duration ) ;
sess - > setPhysMax ( code , 180 ) ;
sess - > setPhysMin ( code , 18 ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( OXI_SPO2 , es . label ) ) {
code = OXI_SPO2 ;
2014-05-04 18:02:41 +00:00
es . physical_minimum = 60 ;
ToTimeDelta ( sess , edf , es , code , recs , duration ) ;
sess - > setPhysMax ( code , 100 ) ;
sess - > setPhysMin ( code , 60 ) ;
2014-09-01 11:39:38 +00:00
} else if ( es . label ! = " Crc16 " ) {
2014-05-04 18:02:41 +00:00
qDebug ( ) < < " Unobserved ResMed SAD Signal " < < es . label ;
}
2011-08-06 13:37:06 +00:00
}
2014-04-17 05:58:57 +00:00
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
timeMutex . lock ( ) ;
timeInLoadSAD + = time . elapsed ( ) ;
timeInEDFOpen + = edfopentime ;
2019-07-31 18:36:40 +00:00
timeInEDFInfo + = edfparsetime ;
2018-05-06 16:59:50 +00:00
timeMutex . unlock ( ) ;
# endif
2011-07-01 10:10:44 +00:00
return true ;
2011-06-29 14:19:38 +00:00
}
2011-06-29 20:30:23 +00:00
2014-05-20 11:51:47 +00:00
bool ResmedLoader : : LoadPLD ( Session * sess , const QString & path )
2011-06-29 14:19:38 +00:00
{
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
QTime time ;
time . start ( ) ;
# endif
2020-05-07 20:43:52 +00:00
QString filename = path . section ( - 2 , - 1 ) ;
2021-09-22 17:52:02 +00:00
// qDebug() << "LoadPLD opening" << filename.section("/", -2, -1);
2019-07-31 18:36:40 +00:00
ResMedEDFInfo edf ;
2020-01-31 00:54:53 +00:00
if ( ! edf . Open ( path ) ) {
2020-05-15 23:58:11 +00:00
qDebug ( ) < < " LoadPLD failed to open " < < filename . section ( " / " , - 2 , - 1 ) ;
2020-02-02 19:29:23 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfopentime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2020-01-31 00:54:53 +00:00
if ( ! edf . Parse ( ) ) {
2020-05-15 23:58:11 +00:00
# ifdef EDF_DEBUG
qDebug ( ) < < " LoadPLD failed to parse " < < filename . section ( " / " , - 2 , - 1 ) ;
# endif
2014-05-20 11:51:47 +00:00
return false ;
2020-01-31 00:54:53 +00:00
}
2018-05-06 16:59:50 +00:00
# ifdef DEBUG_EFFICIENCY
int edfparsetime = time . elapsed ( ) ;
time . start ( ) ;
# endif
2014-05-20 11:51:47 +00:00
2019-04-05 18:00:37 +00:00
// Is it safe to assume the order does not change here?
2014-04-17 05:58:57 +00:00
enum PLDType { MaskPres = 0 , TherapyPres , ExpPress , Leak , RR , Vt , Mv , SnoreIndex , FFLIndex , U1 , U2 } ;
2011-07-01 02:52:02 +00:00
2019-07-31 18:36:40 +00:00
qint64 duration = edf . GetNumDataRecords ( ) * edf . GetDurationMillis ( ) ;
2011-09-17 12:39:00 +00:00
sess - > updateFirst ( edf . startdate ) ;
2014-04-17 05:58:57 +00:00
sess - > updateLast ( edf . startdate + duration ) ;
2011-06-29 17:58:28 +00:00
QString t ;
2014-04-17 05:58:57 +00:00
int emptycnt = 0 ;
2014-04-23 13:19:56 +00:00
EventList * a = nullptr ;
2022-01-09 22:35:32 +00:00
// double rate;
2021-05-02 12:05:49 +00:00
long samples ;
2011-08-01 08:53:26 +00:00
ChannelID code ;
2021-05-02 12:05:49 +00:00
bool square = AppSetting - > squareWavePlots ( ) ;
2020-03-03 21:47:55 +00:00
// The following is a hack to skip the multiple uses of Ti and Te by Resmed for signal labels
// It should be replaced when code in resmed_info class changes the labels to be unique
bool found_Ti_code = false ;
bool found_Te_code = false ;
2014-04-17 05:58:57 +00:00
2021-04-02 23:01:24 +00:00
QDateTime sessionStartDT = QDateTime : : fromMSecsSinceEpoch ( sess - > first ( ) ) ;
2021-05-02 12:05:49 +00:00
// bool forceDebug = (sessionStartDT > QDateTime::fromString("2021-02-26 12:00:00", "yyyy-MM-dd HH:mm:ss")) &&
// (sessionStartDT < QDateTime::fromString("2021-02-27 12:00:00", "yyyy-MM-dd HH:mm:ss"));
bool forceDebug = false ;
2021-04-02 23:01:24 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & es : edf . edfsignals ) {
2014-04-23 13:19:56 +00:00
a = nullptr ;
2021-05-02 12:05:49 +00:00
samples = es . sampleCnt * edf . GetNumDataRecords ( ) ;
2014-04-17 05:58:57 +00:00
2021-05-02 12:05:49 +00:00
if ( samples < = 0 )
2019-07-22 21:01:47 +00:00
continue ;
2014-04-17 05:58:57 +00:00
2022-01-09 22:35:32 +00:00
// rate = double(duration) / double(samples);
2014-04-17 05:58:57 +00:00
2011-07-27 09:21:53 +00:00
//qDebug() << "EVE:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum << es.gain;
2021-04-02 23:01:24 +00:00
if ( forceDebug ) {
qDebug ( ) < < " Session " < < sessionStartDT . toString ( ) < < filename . section ( " / " , - 2 , - 1 ) < < " signal " < < es . label ;
2021-05-02 12:05:49 +00:00
qDebug ( ) < < " \t Second/rec: " < < edf . GetDurationMillis ( ) / 1000 < < " Samples/rec: " < < es . sampleCnt ;
2021-04-02 23:01:24 +00:00
}
2014-05-13 02:56:51 +00:00
if ( matchSignal ( CPAP_Snore , es . label ) ) {
code = CPAP_Snore ;
2021-05-02 12:05:49 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_Pressure , es . label ) ) {
code = CPAP_Pressure ;
2021-07-13 18:36:31 +00:00
// es.physical_maximum = 25;
// es.physical_minimum = 4;
2022-04-06 16:23:43 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , true ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_IPAP , es . label ) ) {
code = CPAP_IPAP ;
2021-07-13 18:36:31 +00:00
// es.physical_maximum = 25;
// es.physical_minimum = 4;
2022-04-06 16:23:43 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , true ) ;
2014-09-01 11:39:38 +00:00
} else if ( matchSignal ( CPAP_EPAP , es . label ) ) { // Expiratory Pressure
code = CPAP_EPAP ;
2021-07-13 18:36:31 +00:00
// es.physical_maximum = 25;
// es.physical_minimum = 4;
2014-09-01 11:39:38 +00:00
2022-04-06 16:23:43 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , true ) ;
2014-09-01 11:39:38 +00:00
} else if ( matchSignal ( CPAP_MinuteVent , es . label ) ) {
2014-05-13 02:56:51 +00:00
code = CPAP_MinuteVent ;
2021-05-02 12:05:49 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_RespRate , es . label ) ) {
code = CPAP_RespRate ;
2022-01-09 22:35:32 +00:00
// a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
// a->AddWaveform(edf.startdate, es.dataArray, samples, duration);
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_TidalVolume , es . label ) ) {
code = CPAP_TidalVolume ;
2022-01-02 01:49:40 +00:00
es . physical_dimension = " mL " ;
2014-04-17 05:58:57 +00:00
es . gain * = 1000.0 ;
es . physical_maximum * = 1000.0 ;
es . physical_minimum * = 1000.0 ;
2019-07-22 21:01:47 +00:00
// es.digital_maximum*=1000.0;
// es.digital_minimum*=1000.0;
2021-05-02 12:05:49 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_Leak , es . label ) ) {
code = CPAP_Leak ;
2014-05-06 02:19:26 +00:00
es . gain * = 60.0 ;
es . physical_maximum * = 60.0 ;
es . physical_minimum * = 60.0 ;
2019-07-22 21:01:47 +00:00
// es.digital_maximum*=60.0;
// es.digital_minimum*=60.0;
2014-04-17 05:58:57 +00:00
es . physical_dimension = " L/M " ;
2021-05-02 12:05:49 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , true ) ;
2014-04-17 05:58:57 +00:00
sess - > setPhysMax ( code , 120.0 ) ;
sess - > setPhysMin ( code , 0 ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_FLG , es . label ) ) {
code = CPAP_FLG ;
2021-05-02 12:05:49 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_MaskPressure , es . label ) ) {
code = CPAP_MaskPressure ;
2021-07-13 18:36:31 +00:00
// es.physical_maximum = 25;
// es.physical_minimum = 4;
2013-10-25 10:39:30 +00:00
2021-05-02 12:05:49 +00:00
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_IE , es . label ) ) { //I:E ratio
code = CPAP_IE ;
2022-11-19 09:02:02 +00:00
es . gain / = 100.0 ;
es . physical_maximum / = 100.0 ;
es . physical_minimum / = 100.0 ;
2019-04-05 18:00:37 +00:00
// qDebug() << "IE Gain, Max, Min" << es.gain << es.physical_maximum << es.physical_minimum;
2019-08-03 14:59:08 +00:00
// qDebug() << "IE count, data..." << es.sampleCnt << es.dataArray[0] << es.dataArray[1] << es.dataArray[2] << es.dataArray[3] << es.dataArray[4];
2022-01-09 22:35:32 +00:00
// a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
// a->AddWaveform(edf.startdate, es.dataArray, samples, duration);
2023-06-19 13:55:49 +00:00
// Fix ToTimeDelta to store inverse of edf data - also fix labels and tool tip
// ToTimeDelta(sess,edf,es, code,samples,duration,0,0, square);
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_Ti , es . label ) ) {
code = CPAP_Ti ;
2020-03-03 21:47:55 +00:00
// There are TWO of these with the same label on 36037, 36039, 36377 and others
// Also 37051 has R5Ti.2s and Ti.2s. We use R5Ti.2s and ignore the Ti.2s
if ( found_Ti_code )
2014-08-22 11:54:17 +00:00
continue ;
2020-03-03 21:47:55 +00:00
found_Ti_code = true ;
2022-01-09 22:35:32 +00:00
// a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
// a->AddWaveform(edf.startdate, es.dataArray, samples, duration);
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_Te , es . label ) ) {
code = CPAP_Te ;
2014-08-22 11:54:17 +00:00
// There are TWO of these with the same label on my VPAP Adapt 36037
2020-03-03 21:47:55 +00:00
if ( found_Te_code )
2014-08-22 11:54:17 +00:00
continue ;
2020-03-03 21:47:55 +00:00
found_Te_code = true ;
2022-01-09 22:35:32 +00:00
// a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
// a->AddWaveform(edf.startdate, es.dataArray, samples, duration);
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2014-05-13 02:56:51 +00:00
} else if ( matchSignal ( CPAP_TgMV , es . label ) ) {
code = CPAP_TgMV ;
2022-01-09 22:35:32 +00:00
// a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
// a->AddWaveform(edf.startdate, es.dataArray, samples, duration);
ToTimeDelta ( sess , edf , es , code , samples , duration , 0 , 0 , square ) ;
2020-03-23 18:41:31 +00:00
} else if ( es . label = = " Va " ) { // Signal used in 36039... What to do with it???
a = nullptr ; // We'll skip it for now
2021-09-12 01:33:01 +00:00
} else if ( es . label = = " AlvMinVent.2s " ) { // Signal used in 28509... What to do with it???
a = nullptr ; // We'll skip it for now
} else if ( es . label = = " CLRatio.2s " ) { // Signal used in 28509... What to do with it???
a = nullptr ; // We'll skip it for now
} else if ( es . label = = " TRRatio.2s " ) { // Signal used in 28509... What to do with it???
a = nullptr ; // We'll skip it for now
2014-08-21 16:13:22 +00:00
} else if ( es . label = = " " ) { // What the hell resmed??
2020-03-23 18:41:31 +00:00
// these empty lables should be changed in resmed_EDFInfo to something unique
2014-04-17 05:58:57 +00:00
if ( emptycnt = = 0 ) {
code = RMS9_E01 ;
2021-05-02 12:05:49 +00:00
// ToTimeDelta(sess, edf, es, code, samples, duration, 0, 0, square);
2014-04-17 05:58:57 +00:00
} else if ( emptycnt = = 1 ) {
code = RMS9_E02 ;
2021-05-02 12:05:49 +00:00
// ToTimeDelta(sess, edf, es, code, samples, duration, 0, 0, square);
2011-07-21 03:35:59 +00:00
} else {
2011-07-27 09:21:53 +00:00
qDebug ( ) < < " Unobserved Empty Signal " < < es . label ;
2011-07-21 03:35:59 +00:00
}
2014-04-17 05:58:57 +00:00
2011-07-21 03:35:59 +00:00
emptycnt + + ;
2014-09-01 11:39:38 +00:00
} else if ( es . label ! = " Crc16 " ) {
2011-08-09 23:44:36 +00:00
qDebug ( ) < < " Unobserved ResMed PLD Signal " < < es . label ;
2014-04-23 13:19:56 +00:00
a = nullptr ;
2011-08-01 08:53:26 +00:00
}
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
if ( a ) {
2020-05-15 01:10:03 +00:00
sess - > updateMin ( code , a - > Min ( ) ) ;
sess - > updateMax ( code , a - > Max ( ) ) ;
2019-12-24 02:44:10 +00:00
sess - > setPhysMin ( code , es . physical_minimum ) ;
sess - > setPhysMax ( code , es . physical_maximum ) ;
a - > setDimension ( es . physical_dimension ) ;
}
}
2020-01-31 00:54:53 +00:00
2019-12-24 02:44:10 +00:00
# ifdef DEBUG_EFFICIENCY
timeMutex . lock ( ) ;
timeInLoadPLD + = time . elapsed ( ) ;
timeInEDFOpen + = edfopentime ;
timeInEDFInfo + = edfparsetime ;
timeMutex . unlock ( ) ;
# endif
return true ;
}
// Convert EDFSignal data to OSCAR's Time-Delta Event format
2021-05-02 12:05:49 +00:00
EventList * buildEventList ( EventStoreType est , EventDataType t_min , EventDataType t_max , EDFSignal & es ,
2019-12-24 02:44:10 +00:00
EventDataType * min , EventDataType * max , double tt , EventList * el , Session * sess , ChannelID code ) ; // forward
void ResmedLoader : : ToTimeDelta ( Session * sess , ResMedEDFInfo & edf , EDFSignal & es , ChannelID code ,
2021-05-02 12:05:49 +00:00
long samples , qint64 duration , EventDataType t_min , EventDataType t_max , bool square )
2019-12-24 02:44:10 +00:00
{
2021-04-27 00:51:48 +00:00
using namespace schema ;
2021-05-02 12:05:49 +00:00
ChannelList channel ;
2021-04-27 00:51:48 +00:00
2021-05-02 12:05:49 +00:00
// QDateTime sessionStartDT = QDateTime:: fromMSecsSinceEpoch(sess->first());
// bool forceDebug = (sessionStartDT > QDateTime::fromString("2021-02-26 12:00:00", "yyyy-MM-dd HH:mm:ss")) &&
// (sessionStartDT < QDateTime::fromString("2021-02-27 12:00:00", "yyyy-MM-dd HH:mm:ss"));
bool forceDebug = false ;
2021-04-27 00:51:48 +00:00
2019-12-24 02:44:10 +00:00
if ( t_min = = t_max ) {
t_min = es . physical_minimum ;
t_max = es . physical_maximum ;
}
# ifdef DEBUG_EFFICIENCY
QElapsedTimer time ;
time . start ( ) ;
# endif
2021-05-02 12:05:49 +00:00
double rate = ( duration / samples ) ; // milliseconds per record
2019-12-24 02:44:10 +00:00
double tt = edf . startdate ;
EventStoreType c = 0 , last ;
int startpos = 0 ;
2021-04-24 20:52:25 +00:00
// There's no reason to skip the first 40 seconds of slow data
2021-10-28 00:47:20 +00:00
// Reduce that to 10 seconds, to allow presssures to stabilise
if ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) {
startpos = 5 ; // Shave the first 10 seconds of pressure data
tt + = rate * startpos ;
}
2022-02-27 16:50:10 +00:00
// Likewise for the values that the device computes for us, but 20 seconds
2022-11-19 09:02:02 +00:00
if ( ( code = = CPAP_MinuteVent ) | | ( code = = CPAP_RespRate ) | | ( code = = CPAP_TidalVolume ) | |
( code = = CPAP_Ti ) | | ( code = = CPAP_Te ) | | ( code = = CPAP_IE ) ) {
2022-01-09 18:49:53 +00:00
startpos = 10 ; // Shave the first 20 seconds of computed data
tt + = rate * startpos ;
}
2019-12-24 02:44:10 +00:00
qint16 * sptr = es . dataArray ;
2021-05-02 12:05:49 +00:00
qint16 * eptr = sptr + samples ;
2019-12-24 02:44:10 +00:00
sptr + = startpos ;
EventDataType min = t_max , max = t_min , tmp ;
EventList * el = nullptr ;
2021-05-02 12:05:49 +00:00
if ( forceDebug )
qDebug ( ) < < " Code: " < < QString : : number ( code , 16 ) < < " Samples: " < < samples ;
if ( samples > startpos + 1 ) {
2019-12-24 02:44:10 +00:00
// Prime last with a good starting value
do {
last = * sptr + + ;
tmp = EventDataType ( last ) * es . gain ;
if ( ( tmp > = t_min ) & & ( tmp < = t_max ) ) {
min = tmp ;
max = tmp ;
el = sess - > AddEventList ( code , EVL_Event , es . gain , es . offset , 0 , 0 ) ;
2021-04-27 00:51:48 +00:00
if ( forceDebug )
2021-05-02 12:05:49 +00:00
// qDebug() << "New EventList:" << channel.channels[code]->code() << QDateTime::fromMSecsSinceEpoch(tt).toString();
qDebug ( ) < < " New EventList: " < < QString : : number ( code , 16 ) < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) ;
2022-12-17 16:40:59 +00:00
2021-05-02 12:05:49 +00:00
el - > AddEvent ( tt , last ) ;
2022-12-17 16:40:59 +00:00
2021-05-02 12:05:49 +00:00
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
qDebug ( ) < < " First Event: " < < tmp < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) < < " Pos: " < < ( sptr - 1 ) - es . dataArray ;
2019-12-24 02:44:10 +00:00
tt + = rate ;
break ;
}
tt + = rate ;
} while ( sptr < eptr ) ;
2021-04-02 23:01:24 +00:00
if ( ! el ) {
2021-04-27 00:51:48 +00:00
qWarning ( ) < < " No eventList for " < < QDateTime : : fromMSecsSinceEpoch ( sess - > first ( ) ) . toString ( ) < < " code "
2021-05-02 12:05:49 +00:00
// << channel.channels[code]->code();
< < QString : : number ( code , 16 ) ;
2021-04-02 23:01:24 +00:00
# ifdef DEBUG_EFFICIENCY
timeMutex . lock ( ) ;
timeInTimeDelta + = time . elapsed ( ) ;
timeMutex . unlock ( ) ;
# endif
2019-12-24 02:44:10 +00:00
return ;
2021-04-02 23:01:24 +00:00
}
2019-12-24 02:44:10 +00:00
2021-05-02 12:05:49 +00:00
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
2021-07-13 18:36:31 +00:00
qDebug ( ) < < " Before loop to buildEventList " < < el - > count ( ) < < " Last: " < < last * es . gain < < " Next: " < < ( * sptr ) * es . gain < <
" Pos: " < < sptr - es . dataArray < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) ;
2019-12-24 02:44:10 +00:00
for ( ; sptr < eptr ; sptr + + ) {
c = * sptr ;
if ( last ! = c ) {
if ( square ) {
2021-05-02 12:05:49 +00:00
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
qDebug ( ) < < " Before square call to buildEventList " < < el - > count ( ) ;
el = buildEventList ( last , t_min , t_max , es , & min , & max , tt , el , sess , code ) ;
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
qDebug ( ) < < " After square call to buildEventList " < < el - > count ( ) ;
2019-12-24 02:44:10 +00:00
}
2021-05-02 12:05:49 +00:00
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
qDebug ( ) < < " Before call to buildEventList " < < el - > count ( ) < < " Cur: " < < c * es . gain < < " Last: " < < last * es . gain < < " Pos: " < < sptr - es . dataArray < <
QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) ;
el = buildEventList ( c , t_min , t_max , es , & min , & max , tt , el , sess , code ) ;
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
qDebug ( ) < < " After call to buildEventList " < < el - > count ( ) ;
2019-12-24 02:44:10 +00:00
}
tt + = rate ;
last = c ;
}
tmp = EventDataType ( c ) * es . gain ;
2021-05-02 12:05:49 +00:00
if ( ( tmp > = t_min ) & & ( tmp < = t_max ) ) {
2019-12-24 02:44:10 +00:00
el - > AddEvent ( tt , c ) ;
2021-05-02 12:05:49 +00:00
if ( forceDebug & & ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) )
qDebug ( ) < < " Last Event: " < < tmp < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) < < " Pos: " < < ( sptr - 1 ) - es . dataArray ;
2024-09-20 00:08:43 +00:00
} else {
2022-01-03 21:23:11 +00:00
qDebug ( ) < < " Failed to add last event - Code: " < < QString : : number ( code , 16 ) < < " Value: " < < tmp < <
QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) < < " Pos: " < < ( sptr - 1 ) - es . dataArray ;
2024-09-20 00:08:43 +00:00
}
2019-12-24 02:44:10 +00:00
2020-05-15 01:10:03 +00:00
sess - > updateMin ( code , min ) ;
sess - > updateMax ( code , max ) ;
2019-12-24 02:44:10 +00:00
sess - > setPhysMin ( code , es . physical_minimum ) ;
sess - > setPhysMax ( code , es . physical_maximum ) ;
sess - > updateLast ( tt ) ;
2021-04-27 00:51:48 +00:00
if ( forceDebug )
2021-05-02 12:05:49 +00:00
// qDebug() << "EventList:" << channel.channels[code]->code() << QDateTime::fromMSecsSinceEpoch(tt).toString() << "Size" << el->count();
qDebug ( ) < < " EventList: " < < QString : : number ( code , 16 ) < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) < < " Size " < < el - > count ( ) ;
2021-04-02 23:01:24 +00:00
} else {
2021-04-27 00:51:48 +00:00
qWarning ( ) < < " not enough records for EventList " < < QDateTime : : fromMSecsSinceEpoch ( sess - > first ( ) ) . toString ( ) < < " code "
2021-05-02 12:05:49 +00:00
// << channel.channels[code]->code();
< < QString : : number ( code , 16 ) ;
2019-12-24 02:44:10 +00:00
}
# ifdef DEBUG_EFFICIENCY
timeMutex . lock ( ) ;
if ( el ! = nullptr ) {
qint64 t = time . nsecsElapsed ( ) ;
int cnt = el - > count ( ) ;
int bytes = cnt * ( sizeof ( EventStoreType ) + sizeof ( quint32 ) ) ;
2021-05-02 12:05:49 +00:00
int wvbytes = samples * ( sizeof ( EventStoreType ) ) ;
2019-12-24 02:44:10 +00:00
auto it = channel_efficiency . find ( code ) ;
if ( it = = channel_efficiency . end ( ) ) {
channel_efficiency [ code ] = wvbytes - bytes ;
channel_time [ code ] = t ;
} else {
it . value ( ) + = wvbytes - bytes ;
channel_time [ code ] + = t ;
}
}
timeInTimeDelta + = time . elapsed ( ) ;
timeMutex . unlock ( ) ;
# endif
} // end ResMedLoader::ToTimeDelta
2021-05-02 12:05:49 +00:00
EventList * buildEventList ( EventStoreType est , EventDataType t_min , EventDataType t_max , EDFSignal & es ,
2019-12-24 02:44:10 +00:00
EventDataType * min , EventDataType * max , double tt , EventList * el , Session * sess , ChannelID code )
{
2021-05-02 12:05:49 +00:00
using namespace schema ;
ChannelList channel ;
// QDateTime sessionStartDT = QDateTime:: fromMSecsSinceEpoch(sess->first());
// bool forceDebug = (sessionStartDT > QDateTime::fromString("2021-02-26 12:00:00", "yyyy-MM-dd HH:mm:ss")) &&
// (sessionStartDT < QDateTime::fromString("2021-02-27 12:00:00", "yyyy-MM-dd HH:mm:ss"));
bool forceDebug = false ;
2019-12-24 02:44:10 +00:00
EventDataType tmp = EventDataType ( est ) * es . gain ;
if ( ( tmp > = t_min ) & & ( tmp < = t_max ) ) {
if ( tmp < * min )
* min = tmp ;
if ( tmp > * max )
* max = tmp ;
el - > AddEvent ( tt , est ) ;
} else {
2022-01-03 21:23:11 +00:00
// if ( tmp > 0 )
2022-01-02 01:49:40 +00:00
qDebug ( ) < < " Code: " < < QString : : number ( code , 16 ) < < " Value: " < < tmp < < " Out of range: \n \t t_min: " < <
t_min < < " t_max: " < < t_max < < " EL count: " < < el - > count ( ) ;
2019-12-24 02:44:10 +00:00
// Out of bounds value, start a new eventlist
2021-05-02 12:05:49 +00:00
// But first drop a closing value that repeats the last one
el - > AddEvent ( tt , el - > raw ( el - > count ( ) - 1 ) ) ;
2019-12-24 02:44:10 +00:00
if ( el - > count ( ) > 1 ) {
// that should be in session, not the eventlist.. handy for debugging though
el - > setDimension ( es . physical_dimension ) ;
el = sess - > AddEventList ( code , EVL_Event , es . gain , es . offset , 0 , 0 ) ;
2021-05-02 12:05:49 +00:00
if ( forceDebug )
// qDebug() << "New EventList:" << channel.channels[code]->code() << QDateTime::fromMSecsSinceEpoch(tt).toString();
qDebug ( ) < < " New EventList: " < < QString : : number ( code , 16 ) < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) ;
} else {
2019-12-24 02:44:10 +00:00
el - > clear ( ) ; // reuse the object
2021-05-02 12:05:49 +00:00
if ( forceDebug )
// qDebug() << "Clear EventList:" << channel.channels[code]->code() << QDateTime::fromMSecsSinceEpoch(tt).toString();
qDebug ( ) < < " Clear EventList: " < < QString : : number ( code , 16 ) < < QDateTime : : fromMSecsSinceEpoch ( tt ) . toString ( ) ;
}
2019-12-24 02:44:10 +00:00
}
2021-05-02 12:05:49 +00:00
return el ;
2019-12-24 02:44:10 +00:00
}
2014-05-20 11:51:47 +00:00
2019-12-24 02:44:10 +00:00
// Check if given string matches any alternative signal names for this channel
bool matchSignal ( ChannelID ch , const QString & name )
{
auto channames = resmed_codes . find ( ch ) ;
if ( channames = = resmed_codes . end ( ) ) {
return false ;
2011-06-29 17:58:28 +00:00
}
2014-04-17 05:58:57 +00:00
2019-12-24 02:44:10 +00:00
for ( auto & string : channames . value ( ) ) {
// Using starts with, because ResMed is very lazy about consistency
if ( name . startsWith ( string , Qt : : CaseInsensitive ) ) {
return true ;
}
}
return false ;
2011-06-29 14:19:38 +00:00
}
2011-06-28 02:21:38 +00:00
2019-12-24 02:44:10 +00:00
void setupResMedTranslationMap ( )
2018-06-07 22:03:20 +00:00
{
2014-04-26 09:54:08 +00:00
////////////////////////////////////////////////////////////////////////////
2022-02-27 16:50:10 +00:00
// Translation lookup table for non-english devices
2021-07-08 18:27:22 +00:00
// Also combine S9, AS10, and AS11 variants
2014-04-26 09:54:08 +00:00
////////////////////////////////////////////////////////////////////////////
2014-05-04 18:02:41 +00:00
// Only put the first part, enough to be identifiable, because ResMed likes
2019-07-22 21:01:47 +00:00
// to crop short the signal names
2014-05-06 17:39:05 +00:00
// Read this from a table?
2014-08-06 14:06:44 +00:00
resmed_codes . clear ( ) ;
2014-05-06 17:39:05 +00:00
2014-09-01 11:39:38 +00:00
// BRP file
2018-06-07 01:53:20 +00:00
resmed_codes [ CPAP_FlowRate ] = QStringList { " Flow " , " Flow.40ms " } ;
resmed_codes [ CPAP_MaskPressureHi ] = QStringList { " Mask Pres " , " Press.40ms " } ;
2020-02-22 02:41:00 +00:00
// resmed_codes[CPAP_TriggerEvent] = QStringList{ "TrigCycEvt.40ms" }; // AC10 VAuto and -S
resmed_codes [ CPAP_RespEvent ] = QStringList { " Resp Event " , " TrigCycEvt.40ms " } ; // S9 VPAPS and STA-IVAPS call it RespEvent
2014-09-01 11:39:38 +00:00
// PLD File
2018-06-07 01:53:20 +00:00
resmed_codes [ CPAP_MaskPressure ] = QStringList { " Mask Pres " , " MaskPress.2s " } ;
2020-02-22 02:41:00 +00:00
// resmed_codes[CPAP_RespEvent] = QStringList {"Resp Event" };
2018-06-07 01:53:20 +00:00
resmed_codes [ CPAP_Pressure ] = QStringList { " Therapy Pres " , " Press.2s " } ; // Un problemo... IPAP also uses Press.2s.. check the mode :/
2022-05-08 21:41:53 +00:00
// STR signals
2018-06-07 01:53:20 +00:00
resmed_codes [ CPAP_IPAP ] = QStringList { " Insp Pres " , " IPAP " , " S.BL.IPAP " } ;
resmed_codes [ CPAP_EPAP ] = QStringList { " Exp Pres " , " EprPress.2s " , " EPAP " , " S.BL.EPAP " , " EPRPress.2s " } ;
resmed_codes [ CPAP_EPAPHi ] = QStringList { " Max EPAP " } ;
resmed_codes [ CPAP_EPAPLo ] = QStringList { " Min EPAP " , " S.VA.MinEPAP " } ;
resmed_codes [ CPAP_IPAPHi ] = QStringList { " Max IPAP " , " S.VA.MaxIPAP " } ;
resmed_codes [ CPAP_IPAPLo ] = QStringList { " Min IPAP " } ;
resmed_codes [ CPAP_PS ] = QStringList { " PS " , " S.VA.PS " } ;
resmed_codes [ CPAP_PSMin ] = QStringList { " Min PS " } ;
resmed_codes [ CPAP_PSMax ] = QStringList { " Max PS " } ;
2021-05-23 22:00:19 +00:00
resmed_codes [ CPAP_Leak ] = QStringList { " Leak " , " Leck " , " Fuites " , " Fuite " , " Fuga " , " \xE6 \xBC \x8F \xE6 \xB0 \x94 " , " Lekk " , " Läck " , " Läck " , " Leak.2s " , " Sı zı ntı " } ;
2018-06-07 01:53:20 +00:00
resmed_codes [ CPAP_RespRate ] = QStringList { " RR " , " AF " , " FR " , " RespRate.2s " } ;
resmed_codes [ CPAP_MinuteVent ] = QStringList { " MV " , " VM " , " MinVent.2s " } ;
resmed_codes [ CPAP_TidalVolume ] = QStringList { " Vt " , " VC " , " TidVol.2s " } ;
resmed_codes [ CPAP_IE ] = QStringList { " I:E " , " IERatio.2s " } ;
resmed_codes [ CPAP_Snore ] = QStringList { " Snore " , " Snore.2s " } ;
resmed_codes [ CPAP_FLG ] = QStringList { " FFL Index " , " FlowLim.2s " } ;
resmed_codes [ CPAP_Ti ] = QStringList { " Ti " , " B5ITime.2s " } ;
resmed_codes [ CPAP_Te ] = QStringList { " Te " , " B5ETime.2s " } ;
resmed_codes [ CPAP_TgMV ] = QStringList { " TgMV " , " TgtVent.2s " } ;
2021-05-22 19:15:52 +00:00
resmed_codes [ OXI_Pulse ] = QStringList { " Pulse " , " Puls " , " Pouls " , " Pols " , " Pulse.1s " , " Nabiz " } ;
2018-06-07 01:53:20 +00:00
resmed_codes [ OXI_SPO2 ] = QStringList { " SpO2 " , " SpO2.1s " } ;
resmed_codes [ CPAP_Obstructive ] = QStringList { " Obstructive apnea " } ;
resmed_codes [ CPAP_Hypopnea ] = QStringList { " Hypopnea " } ;
resmed_codes [ CPAP_Apnea ] = QStringList { " Apnea " } ;
resmed_codes [ CPAP_RERA ] = QStringList { " Arousal " } ;
resmed_codes [ CPAP_ClearAirway ] = QStringList { " Central apnea " } ;
2021-05-22 18:54:52 +00:00
resmed_codes [ CPAP_Mode ] = QStringList { " Mode " , " Modus " , " Funktion " , " \xE6 \xA8 \xA1 \xE5 \xBC \x8F " , " Mod " } ;
resmed_codes [ RMS9_SetPressure ] = QStringList { " Set Pressure " , " Eingest. Druck " , " Ingestelde druk " , " \xE8 \xAE \xBE \xE5 \xAE \x9A \xE5 \x8E \x8B \xE5 \x8A \x9B " , " Pres. prescrite " , " Inställt tryck " , " Inställt tryck " , " S.C.Press " , " Bası ncı Ayarl " } ;
2018-06-07 01:53:20 +00:00
resmed_codes [ RMS9_EPR ] = QStringList { " EPR " , " \xE5 \x91 \xBC \xE6 \xB0 \x94 \xE9 \x87 \x8A \xE5 \x8E \x8B \x28 \x45 \x50 " } ;
2021-05-22 18:54:52 +00:00
resmed_codes [ RMS9_EPRLevel ] = QStringList { " EPR Level " , " EPR-Stufe " , " EPR-niveau " , " \x45 \x50 \x52 \x20 \xE6 \xB0 \xB4 \xE5 \xB9 \xB3 " , " Niveau EPR " , " EPR-nivå " , " EPR-nivÃ¥ " , " S.EPR.Level " , " EPR Düzeyi " } ;
2021-07-08 18:27:22 +00:00
resmed_codes [ CPAP_PressureMax ] = QStringList { " Max Pressure " , " Max. Druck " , " Max druk " , " \xE6 \x9C \x80 \xE5 \xA4 \xA7 \xE5 \x8E \x8B \xE5 \x8A \x9B " , " Pression max. " , " Max tryck " , " S.AS.MaxPress " , " S.A.MaxPress " , " Azami Bası nç " } ;
resmed_codes [ CPAP_PressureMin ] = QStringList { " Min Pressure " , " Min. Druck " , " Min druk " , " \xE6 \x9C \x80 \xE5 \xB0 \x8F \xE5 \x8E \x8B \xE5 \x8A \x9B " , " Pression min. " , " Min tryck " , " S.AS.MinPress " , " S.A.MinPress " , " Min Bası nç " } ;
2014-09-01 04:49:05 +00:00
2018-06-07 01:53:20 +00:00
//resmed_codes[RMS9_EPR].push_back("S.EPR.EPRType");
2011-09-23 03:54:48 +00:00
}
2011-06-28 02:21:38 +00:00
2014-08-06 14:06:44 +00:00
2019-12-24 02:44:10 +00:00
// don't really need this anymore, but perhaps it's useful info for reference
// Resmed_Model_Map = {
// { "S9 Escape", { 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301, 36361 } },
// { "S9 Escape Auto", { 36002, 36012, 36022, 36302, 36362 } },
// { "S9 Elite", { 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203, 36223, 36243, 36263, 36303, 36343, 36363 } },
// { "S9 Autoset", { 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205, 36225, 36245, 36265, 36305, 36325, 36345, 36365 } },
// { "S9 AutoSet CS", { 36100, 36110, 36120, 36140, 36200, 36220, 36360 } },
// { "S9 AutoSet 25", { 36106, 36116, 36126, 36146, 36206, 36226, 36366 } },
// { "S9 AutoSet for Her", { 36065 } },
// { "S9 VPAP S", { 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224, 36284, 36304 } },
// { "S9 VPAP Auto", { 36006, 36016, 36026 } },
// { "S9 VPAP Adapt", { 36037, 36007, 36017, 36027, 36367 } },
// { "S9 VPAP ST", { 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 } },
// { "S9 VPAP ST 22", { 36118, 36128 } },
// { "S9 VPAP ST-A", { 36039, 36159, 36169, 36379 } },
// //S8 Series
// { "S8 Escape", { 33007 } },
// { "S8 Elite II", { 33039 } },
// { "S8 Escape II", { 33051 } },
// { "S8 Escape II AutoSet", { 33064 } },
// { "S8 AutoSet II", { 33129 } },
// };
//
// Return the model name matching the supplied model number.
// const QString & lookupModel(quint16 model)
// {
//
// for (auto it=Resmed_Model_Map.begin(),end = Resmed_Model_Map.end(); it != end; ++it) {
// QList<quint16> & list = it.value();
// for (auto val : list) {
// if (val == model) {
// return it.key();
// }
// }
// }
// return STR_UnknownModel;
// }
2011-06-28 02:21:38 +00:00
2014-05-04 18:02:41 +00:00
////////////////////////////////////////////////////////////////////////////////////////////////
// Model number information
// 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203,
// 36223, 36243, 36263, 36303, 36343, 36363 S9 Elite Series
// 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205,
// 36225, 36245, 36265, 36305, 36325, 36345, 36365 S9 AutoSet Series
// 36065 S9 AutoSet for Her
// 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301,
// 36361 S9 Escape
// 36002, 36012, 36022, 36302, 36362 S9 Escape Auto
// 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224,
// 36284, 36304 S9 VPAP S (+ H5i, + Climate Control)
// 36006, 36016, 36026 S9 VPAP AUTO (+ H5i, + Climate Control)
// 36007, 36017, 36027, 36367
// S9 VPAP ADAPT (+ H5i, + Climate
// Control)
// 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 S9 VPAP ST (+ H5i, + Climate Control)
// 36100, 36110, 36120, 36140, 36200, 36220, 36360 S9 AUTOSET CS
// 36106, 36116, 36126, 36146, 36206, 36226, 36366 S9 AUTOSET 25
// 36118, 36128 S9 VPAP ST 22
// 36039, 36159, 36169, 36379 S9 VPAP ST-A
// 24921, 24923, 24925, 24926, 24927 ResMed Power Station II (RPSII)
// 33030 S8 Compact
// 33001, 33007, 33013, 33036, 33060 S8 Escape
// 33032 S8 Lightweight
// 33033 S8 AutoScore
// 33048, 33051, 33052, 33053, 33054, 33061 S8 Escape II
// 33055 S8 Lightweight II
// 33021 S8 Elite
// 33039, 33045, 33062, 33072, 33073, 33074, 33075 S8 Elite II
// 33044 S8 AutoScore II
// 33105, 33112, 33126 S8 AutoSet (including Spirit & Vantage)
// 33128, 33137 S8 Respond
// 33129, 33141, 33150 S8 AutoSet II
// 33136, 33143, 33144, 33145, 33146, 33147, 33148 S8 AutoSet Spirit II
// 33138 S8 AutoSet C
// 26101, 26121 VPAP Auto 25
// 26119, 26120 VPAP S
// 26110, 26122 VPAP ST
// 26104, 26105, 26125, 26126 S8 Auto 25
// 26102, 26103, 26106, 26107, 26108, 26109, 26123, 26127 VPAP IV
// 26112, 26113, 26114, 26115, 26116, 26117, 26118, 26124 VPAP IV ST