Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
/* SleepLib Resvent Loader Implementation
*
* Copyright ( c ) 2019 - 2023 The OSCAR Team
* Copyright ( c ) 2011 - 2018 Mark Watkins < mark @ jedimark . net >
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of the source code
* for more details . */
//********************************************************************************************
// Please only INCREMENT the resvent_data_version in resvent_loader.h when making changes
// that change loader behaviour or modify channels in a manner that fixes old data imports.
// Note that changing the data version will require a reimport of existing data for which OSCAR
// does not keep a backup - so it should be avoided if possible.
// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
2023-11-14 21:06:48 +00:00
# define TEST_MACROS_ENABLEDoff // Turn off for offical release.
2023-10-30 16:49:13 +00:00
# include <test_macros.h>
# include <QCoreApplication>
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
# include <QString>
# include <QDateTime>
# include <QDir>
# include <QDirIterator>
# include <QFile>
# include <QDebug>
2023-05-13 21:23:22 +00:00
# include <QVector>
# include <QMap>
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
# include <QStringList>
# include <cmath>
# include "resvent_loader.h"
# ifdef DEBUG_EFFICIENCY
# include <QElapsedTimer> // only available in 4.8
# endif
// Files WXX_XX contain flow rate and pressure and PXX_XX contain Pressure, IPAP, EPAP, Leak, Vt, MV, RR, Ti, IE, Spo2, PR
// Both files contain a little header of size 0x24 bytes. In offset 0x12 contain the total number of different records in
// the different files of the same type. And later contain the previous describe quantity of description header of size 0x20
// containing the details for every type of record (e.g. sample chunk size).
ResventLoader : : ResventLoader ( )
{
const QString RESVENT_ICON = " :/icons/resvent.png " ;
QString s = newInfo ( ) . series ;
m_pixmap_paths [ s ] = RESVENT_ICON ;
m_pixmaps [ s ] = QPixmap ( RESVENT_ICON ) ;
m_type = MT_CPAP ;
}
ResventLoader : : ~ ResventLoader ( )
{
}
const QString kResventTherapyFolder = " THERAPY " ;
const QString kResventConfigFolder = " CONFIG " ;
const QString kResventRecordFolder = " RECORD " ;
const QString kResventSysConfigFilename = " SYSCFG " ;
constexpr qint64 kDateTimeOffset = 7 * 60 * 60 * 1000 ;
constexpr int kMainHeaderSize = 0x24 ;
constexpr int kDescriptionHeaderSize = 0x20 ;
constexpr int kChunkDurationInSecOffset = 0x10 ;
constexpr int kDescriptionCountOffset = 0x12 ;
constexpr int kDescriptionSamplesByChunk = 0x1e ;
2023-05-13 20:52:03 +00:00
constexpr double kDefaultGain = 0.01 ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
2023-10-30 23:20:20 +00:00
int countDaysWithRecordData ( QDir & dir ) {
int count = 0 ;
QDirIterator it ( dir ) ;
while ( it . hasNext ( ) ) {
QFileInfo fi = it . next ( ) ;
QString filename = fi . fileName ( ) ;
if ( filename = = " . " | | filename = = " .. " ) continue ;
bool isDir = dir . cd ( filename ) ;
if ( isDir ) {
// It is a directory.
2023-11-14 21:06:48 +00:00
// check if name is between 1 and 31. for day in month
int dayInMonth = filename . toInt ( ) ;
if ( dayInMonth > 0 & & dayInMonth < = 31 ) {
// in month folder.
count + + ;
} else {
// recurse into folder.
count + = countDaysWithRecordData ( dir ) ;
}
2023-10-30 23:20:20 +00:00
//Change back to parent.
dir . cdUp ( ) ;
}
}
return count ;
}
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
bool ResventLoader : : Detect ( const QString & givenpath )
{
QDir dir ( givenpath ) ;
if ( ! dir . exists ( ) ) {
return false ;
}
if ( ! dir . exists ( kResventTherapyFolder ) ) {
return false ;
}
dir . cd ( kResventTherapyFolder ) ;
if ( ! dir . exists ( kResventConfigFolder ) ) {
return false ;
}
2023-10-30 23:20:20 +00:00
if ( ! dir . cd ( kResventRecordFolder ) ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
return false ;
}
2023-10-30 23:20:20 +00:00
int count = countDaysWithRecordData ( dir ) ;
progress = 0 ;
emit setProgressMax ( count + 1 ) ;
emit setProgressValue ( progress ) ;
QCoreApplication : : processEvents ( ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
return true ;
}
MachineInfo ResventLoader : : PeekInfo ( const QString & path )
{
if ( ! Detect ( path ) ) {
return MachineInfo ( ) ;
}
MachineInfo info = newInfo ( ) ;
const auto sys_config_path = path + QDir : : separator ( ) + kResventTherapyFolder + QDir : : separator ( ) + kResventConfigFolder + QDir : : separator ( ) + kResventSysConfigFilename ;
if ( ! QFile : : exists ( sys_config_path ) ) {
qDebug ( ) < < " Resvent Data card has no " < < kResventSysConfigFilename < < " file in " < < sys_config_path ;
return MachineInfo ( ) ;
}
QFile f ( sys_config_path ) ;
f . open ( QIODevice : : ReadOnly | QIODevice : : Text ) ;
f . seek ( 4 ) ;
while ( ! f . atEnd ( ) ) {
QString line = f . readLine ( ) . trimmed ( ) ;
const auto elems = line . split ( " = " ) ;
2023-05-10 19:41:27 +00:00
Q_ASSERT ( elems . size ( ) = = 2 ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
if ( elems [ 0 ] = = " models " ) {
info . model = elems [ 1 ] ;
}
else if ( elems [ 0 ] = = " sn " ) {
info . serial = elems [ 1 ] ;
}
else if ( elems [ 0 ] = = " num " ) {
info . version = elems [ 1 ] . toInt ( ) ;
}
else if ( elems [ 0 ] = = " num " ) {
info . type = MachineType : : MT_CPAP ;
}
}
if ( info . model . contains ( " Point " , Qt : : CaseInsensitive ) ) {
info . brand = " Hoffrichter " ;
} else {
info . brand = " Resvent " ;
}
return info ;
}
2023-05-13 21:23:22 +00:00
QVector < QDate > GetSessionsDate ( const QString & dirpath ) {
QVector < QDate > sessions_date ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto records_path = dirpath + QDir : : separator ( ) + kResventTherapyFolder + QDir : : separator ( ) + kResventRecordFolder ;
QDir records_folder ( records_path ) ;
const auto year_month_folders = records_folder . entryList ( QStringList ( ) , QDir : : Dirs | QDir : : NoDotAndDotDot , QDir : : Name ) ;
std : : for_each ( year_month_folders . cbegin ( ) , year_month_folders . cend ( ) , [ & ] ( const QString & year_month_folder_name ) {
if ( year_month_folder_name . length ( ) ! = 6 ) {
return ;
}
const int year = std : : stoi ( year_month_folder_name . left ( 4 ) . toStdString ( ) ) ;
const int month = std : : stoi ( year_month_folder_name . right ( 2 ) . toStdString ( ) ) ;
const auto year_month_folder_path = records_path + QDir : : separator ( ) + year_month_folder_name ;
QDir year_month_folder ( year_month_folder_path ) ;
const auto session_folders = year_month_folder . entryList ( QStringList ( ) , QDir : : Dirs | QDir : : NoDotAndDotDot , QDir : : Name ) ;
std : : for_each ( session_folders . cbegin ( ) , session_folders . cend ( ) , [ & ] ( const QString & day_folder ) {
const auto day = std : : stoi ( day_folder . toStdString ( ) ) ;
sessions_date . push_back ( QDate ( year , month , day ) ) ;
} ) ;
} ) ;
return sessions_date ;
}
enum class EventType {
2023-11-14 21:06:48 +00:00
//UsageSec = 1,
//UnixStart = 2,
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
ObstructiveApnea = 17 ,
CentralApnea = 18 ,
Hypopnea = 19 ,
FlowLimitation = 20 ,
RERA = 21 ,
PeriodicBreathing = 22 ,
Snore = 23
} ;
struct EventData {
EventType type ;
QDateTime date_time ;
int duration ;
} ;
struct UsageData {
QString number { } ;
QDateTime start_time { } ;
QDateTime end_time { } ;
qint32 countAHI = 0 ;
qint32 countOAI = 0 ;
qint32 countCAI = 0 ;
qint32 countAI = 0 ;
qint32 countHI = 0 ;
qint32 countRERA = 0 ;
qint32 countSNI = 0 ;
qint32 countBreath = 0 ;
} ;
2023-05-13 21:23:22 +00:00
void UpdateEvents ( EventType event_type , const QMap < EventType , QVector < EventData > > & events , Session * session ) {
static QMap < EventType , unsigned int > mapping { { EventType : : ObstructiveApnea , CPAP_Obstructive } ,
2023-11-11 18:10:48 +00:00
{ EventType : : CentralApnea , CPAP_ClearAirway } ,
2023-05-13 21:23:22 +00:00
{ EventType : : Hypopnea , CPAP_Hypopnea } ,
{ EventType : : FlowLimitation , CPAP_FlowLimit } ,
{ EventType : : RERA , CPAP_RERA } ,
{ EventType : : PeriodicBreathing , CPAP_PB } ,
{ EventType : : Snore , CPAP_Snore } } ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto it_events = events . find ( event_type ) ;
const auto it_mapping = mapping . find ( event_type ) ;
if ( it_events = = events . cend ( ) | | it_mapping = = mapping . cend ( ) ) {
return ;
}
2023-05-13 21:23:22 +00:00
EventList * event_list = session - > AddEventList ( it_mapping . value ( ) , EVL_Event ) ;
std : : for_each ( it_events . value ( ) . cbegin ( ) , it_events . value ( ) . cend ( ) , [ & ] ( const EventData & event_data ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
event_list - > AddEvent ( event_data . date_time . toMSecsSinceEpoch ( ) + kDateTimeOffset , event_data . duration ) ;
} ) ;
}
QString GetSessionFolder ( const QString & dirpath , const QDate & session_date ) {
2023-11-10 07:40:06 +00:00
const auto year_month_folder = QString ( " %1%2 " ) . arg ( session_date . year ( ) ) . arg ( session_date . month ( ) , 2 , 10 , QLatin1Char ( ' 0 ' ) ) ;
const auto day_folder = QString ( " %1 " ) . arg ( session_date . day ( ) , 2 , 10 , QLatin1Char ( ' 0 ' ) ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto session_folder_path = dirpath + QDir : : separator ( ) + kResventTherapyFolder + QDir : : separator ( ) + kResventRecordFolder + QDir : : separator ( ) + year_month_folder + QDir : : separator ( ) + day_folder ;
return session_folder_path ;
}
2023-11-11 18:10:48 +00:00
bool VerifyEvent ( EventData & eventData ) {
switch ( eventData . type ) {
2023-11-14 21:06:48 +00:00
case EventType : : Snore :
case EventType : : FlowLimitation :
case EventType : : PeriodicBreathing :
case EventType : : Hypopnea :
2023-11-11 18:10:48 +00:00
case EventType : : ObstructiveApnea : // OA
case EventType : : CentralApnea : // CA and same clear airway.
// adjust time of event to be after the event ends rather than when the event starts.
eventData . date_time = eventData . date_time . addMSecs ( eventData . duration * 1000 ) ;
break ;
case EventType : : RERA :
eventData . duration = 0 ; // duration is large and suppress duration display of eariler OA events.
break ;
default :
2023-11-14 21:06:48 +00:00
DEBUGFW Q ( ( int ) eventData . type ) O ( eventData . date_time ) Q ( eventData . duration ) ;
2023-11-11 18:10:48 +00:00
break ;
}
return true ;
}
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
void LoadEvents ( const QString & session_folder_path , Session * session , const UsageData & usage ) {
const auto event_file_path = session_folder_path + QDir : : separator ( ) + " EV " + usage . number ;
2023-11-11 18:10:48 +00:00
// Oscar (resmed) plots events at end.
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
2023-05-13 21:23:22 +00:00
QMap < EventType , QVector < EventData > > events ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
QFile f ( event_file_path ) ;
f . open ( QIODevice : : ReadOnly | QIODevice : : Text ) ;
f . seek ( 4 ) ;
while ( ! f . atEnd ( ) ) {
2023-11-11 18:10:48 +00:00
QString line = f . readLine ( ) . trimmed ( ) ; // ID=20,DT=1692022874,DR=6,GD=0,
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
2023-05-10 15:09:22 +00:00
# if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto elems = line . split ( " , " , Qt : : SkipEmptyParts ) ;
2023-05-10 15:09:22 +00:00
# else
const auto elems = line . split ( " , " , QString : : SkipEmptyParts ) ;
# endif
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
if ( elems . size ( ) ! = 4 ) {
continue ;
}
const auto event_type_elems = elems . at ( 0 ) . split ( " = " ) ;
const auto date_time_elems = elems . at ( 1 ) . split ( " = " ) ;
const auto duration_elems = elems . at ( 2 ) . split ( " = " ) ;
2023-05-10 19:41:27 +00:00
Q_ASSERT ( event_type_elems . size ( ) = = 2 ) ;
Q_ASSERT ( date_time_elems . size ( ) = = 2 ) ;
2023-11-11 18:10:48 +00:00
Q_ASSERT ( duration_elems . size ( ) = = 2 ) ;
auto event_type = static_cast < EventType > ( std : : stoi ( event_type_elems [ 1 ] . toStdString ( ) ) ) ;
auto duration = std : : stoi ( duration_elems [ 1 ] . toStdString ( ) ) ;
auto date_time = QDateTime : : fromTime_t ( std : : stoi ( date_time_elems [ 1 ] . toStdString ( ) ) ) ;
2023-11-12 00:30:07 +00:00
2023-11-11 18:10:48 +00:00
EventData eventData ( { event_type , date_time , duration } ) ;
VerifyEvent ( eventData ) ;
events [ event_type ] . push_back ( eventData ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
2023-05-13 21:23:22 +00:00
static QVector < EventType > mapping { EventType : : ObstructiveApnea ,
EventType : : CentralApnea ,
EventType : : Hypopnea ,
EventType : : FlowLimitation ,
EventType : : RERA ,
EventType : : PeriodicBreathing ,
EventType : : Snore } ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
std : : for_each ( mapping . cbegin ( ) , mapping . cend ( ) , [ & ] ( EventType event_type ) {
UpdateEvents ( event_type , events , session ) ;
} ) ;
2023-11-14 23:39:17 +00:00
session - > settings [ CPAP_PressureMin ] = 4.0 ;
2023-11-17 17:48:08 +00:00
session - > settings [ CPAP_PressureMax ] = 20.0 ; // was qMax(session->calcMax(CPAP_Pressure),session->calcMax(CPAP_IPAP)); // this results in a change of device settings.
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
template < typename T >
T read_from_file ( QFile & f ) {
T data { } ;
f . read ( reinterpret_cast < char * > ( & data ) , sizeof ( T ) ) ;
return data ;
}
struct WaveFileData {
unsigned int wave_event_id ;
QString file_base_name ;
unsigned int sample_rate_offset ;
unsigned int start_offset ;
} ;
EventList * GetEventList ( const QString & name , Session * session , float sample_rate = 0.0 ) {
if ( name = = " Press " ) {
2023-05-13 20:52:03 +00:00
return session - > AddEventList ( CPAP_Pressure , EVL_Event ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
else if ( name = = " IPAP " ) {
2023-05-13 20:52:03 +00:00
return session - > AddEventList ( CPAP_IPAP , EVL_Event ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
else if ( name = = " EPAP " ) {
2023-05-13 20:52:03 +00:00
return session - > AddEventList ( CPAP_EPAP , EVL_Event ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
else if ( name = = " Leak " ) {
2023-11-14 21:06:48 +00:00
// was was adjusted from the default 1.0 to 13. so that the graph of gain would match iMatrix values.
return session - > AddEventList ( CPAP_Leak , EVL_Event , 13.0 ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
else if ( name = = " Vt " ) {
return session - > AddEventList ( CPAP_TidalVolume , EVL_Event ) ;
}
else if ( name = = " MV " ) {
return session - > AddEventList ( CPAP_MinuteVent , EVL_Event ) ;
}
else if ( name = = " RR " ) {
return session - > AddEventList ( CPAP_RespRate , EVL_Event ) ;
}
else if ( name = = " Ti " ) {
return session - > AddEventList ( CPAP_Ti , EVL_Event ) ;
}
else if ( name = = " I:E " ) {
return session - > AddEventList ( CPAP_IE , EVL_Event ) ;
}
else if ( name = = " SpO2 " | | name = = " PR " ) {
// Not present
return nullptr ;
}
else if ( name = = " Pressure " ) {
2023-05-13 20:52:03 +00:00
return session - > AddEventList ( CPAP_MaskPressure , EVL_Waveform , kDefaultGain , 0.0 , 0.0 , 0.0 , 1000.0 / sample_rate ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
else if ( name = = " Flow " ) {
2023-05-13 20:52:03 +00:00
return session - > AddEventList ( CPAP_FlowRate , EVL_Waveform , kDefaultGain , 0.0 , 0.0 , 0.0 , 1000.0 / sample_rate ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
else {
// Not supported
2023-05-10 19:41:27 +00:00
Q_ASSERT ( false ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
return nullptr ;
}
}
struct ChunkData {
EventList * event_list ;
uint16_t samples_by_chunk ;
qint64 start_time ;
int total_samples_by_chunk ;
float sample_rate ;
} ;
QString ReadDescriptionName ( QFile & f ) {
constexpr int kNameSize = 9 ;
2023-05-14 10:44:22 +00:00
QVector < char > name ( kNameSize ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto readed = f . read ( name . data ( ) , kNameSize - 1 ) ;
2023-05-10 19:41:27 +00:00
Q_ASSERT ( readed = = kNameSize - 1 ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
return QString ( name . data ( ) ) ;
}
2023-05-13 21:23:22 +00:00
void ReadWaveFormsHeaders ( QFile & f , QVector < ChunkData > & wave_forms , Session * session , const UsageData & usage ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
f . seek ( kChunkDurationInSecOffset ) ;
const auto chunk_duration_in_sec = read_from_file < uint16_t > ( f ) ;
f . seek ( kDescriptionCountOffset ) ;
const auto description_count = read_from_file < uint16_t > ( f ) ;
wave_forms . resize ( description_count ) ;
2023-11-14 21:06:48 +00:00
//DEBUGFW Q(chunk_duration_in_sec) Q(description_count);
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
for ( unsigned int i = 0 ; i < description_count ; i + + ) {
const auto description_header_offset = kMainHeaderSize + i * kDescriptionHeaderSize ;
f . seek ( description_header_offset ) ;
const auto name = ReadDescriptionName ( f ) ;
f . seek ( description_header_offset + kDescriptionSamplesByChunk ) ;
const auto samples_by_chunk = read_from_file < uint16_t > ( f ) ;
wave_forms [ i ] . sample_rate = 1.0 * samples_by_chunk / chunk_duration_in_sec ;
wave_forms [ i ] . event_list = GetEventList ( name , session , wave_forms [ i ] . sample_rate ) ;
wave_forms [ i ] . samples_by_chunk = samples_by_chunk ;
wave_forms [ i ] . start_time = usage . start_time . toMSecsSinceEpoch ( ) ;
2023-11-14 21:06:48 +00:00
DEBUGFW Q ( name )
Q ( samples_by_chunk )
QQ ( " sampleRate " , wave_forms [ i ] . sample_rate )
QQ ( " epoch " , wave_forms [ i ] . start_time )
DATETIME ( wave_forms [ i ] . start_time )
;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
}
void LoadOtherWaveForms ( const QString & session_folder_path , Session * session , const UsageData & usage ) {
QDir session_folder ( session_folder_path ) ;
const auto wave_files = session_folder . entryList ( QStringList ( ) < < " P " + usage . number + " _* " , QDir : : Files , QDir : : Name ) ;
2023-05-13 21:23:22 +00:00
QVector < ChunkData > wave_forms ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
bool initialized = false ;
std : : for_each ( wave_files . cbegin ( ) , wave_files . cend ( ) , [ & ] ( const QString & wave_file ) {
// W01_ file
QFile f ( session_folder_path + QDir : : separator ( ) + wave_file ) ;
f . open ( QIODevice : : ReadOnly ) ;
if ( ! initialized ) {
ReadWaveFormsHeaders ( f , wave_forms , session , usage ) ;
initialized = true ;
}
f . seek ( kMainHeaderSize + wave_forms . size ( ) * kDescriptionHeaderSize ) ;
std : : vector < qint16 > chunk ( std : : max_element ( wave_forms . cbegin ( ) , wave_forms . cend ( ) , [ ] ( const ChunkData & lhs , const ChunkData & rhs ) {
return lhs . samples_by_chunk < rhs . samples_by_chunk ;
} ) - > samples_by_chunk ) ;
while ( ! f . atEnd ( ) ) {
2023-05-13 21:23:22 +00:00
for ( int i = 0 ; i < wave_forms . size ( ) ; i + + ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto & wave_form = wave_forms [ i ] . event_list ;
const auto samples_by_chunk_actual = wave_forms [ i ] . samples_by_chunk ;
auto & start_time_current = wave_forms [ i ] . start_time ;
auto & total_samples_by_chunk = wave_forms [ i ] . total_samples_by_chunk ;
const auto sample_rate = wave_forms [ i ] . sample_rate ;
2023-11-11 18:10:48 +00:00
const auto duration = samples_by_chunk_actual * 1000.0 / sample_rate ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto readed = f . read ( reinterpret_cast < char * > ( chunk . data ( ) ) , chunk . size ( ) * sizeof ( qint16 ) ) ;
if ( wave_form ) {
const auto readed_elements = readed / sizeof ( qint16 ) ;
if ( readed_elements ! = samples_by_chunk_actual ) {
std : : fill ( std : : begin ( chunk ) + readed_elements , std : : end ( chunk ) , 0 ) ;
}
int offset = 0 ;
std : : for_each ( chunk . cbegin ( ) , chunk . cend ( ) , [ & ] ( const qint16 & value ) {
2023-05-13 20:52:03 +00:00
wave_form - > AddEvent ( start_time_current + offset + kDateTimeOffset , value * kDefaultGain ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
offset + = 1000.0 / sample_rate ;
} ) ;
}
2023-11-11 18:10:48 +00:00
start_time_current + = duration ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
total_samples_by_chunk + = samples_by_chunk_actual ;
}
}
} ) ;
2023-05-13 21:23:22 +00:00
QVector < qint16 > chunk ;
for ( int i = 0 ; i < wave_forms . size ( ) ; i + + ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto & wave_form = wave_forms [ i ] ;
const auto expected_samples = usage . start_time . msecsTo ( usage . end_time ) / 1000.0 * wave_form . sample_rate ;
if ( wave_form . total_samples_by_chunk < expected_samples ) {
chunk . resize ( expected_samples - wave_form . total_samples_by_chunk ) ;
if ( wave_form . event_list ) {
int offset = 0 ;
std : : for_each ( chunk . cbegin ( ) , chunk . cend ( ) , [ & ] ( const qint16 & value ) {
2023-05-13 20:52:03 +00:00
wave_form . event_list - > AddEvent ( wave_form . start_time + offset + kDateTimeOffset , value * kDefaultGain ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
offset + = 1000.0 / wave_form . sample_rate ;
} ) ;
}
}
}
}
void LoadWaveForms ( const QString & session_folder_path , Session * session , const UsageData & usage ) {
QDir session_folder ( session_folder_path ) ;
const auto wave_files = session_folder . entryList ( QStringList ( ) < < " W " + usage . number + " _* " , QDir : : Files , QDir : : Name ) ;
2023-05-13 21:23:22 +00:00
QVector < ChunkData > wave_forms ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
bool initialized = false ;
std : : for_each ( wave_files . cbegin ( ) , wave_files . cend ( ) , [ & ] ( const QString & wave_file ) {
// W01_ file
QFile f ( session_folder_path + QDir : : separator ( ) + wave_file ) ;
f . open ( QIODevice : : ReadOnly ) ;
if ( ! initialized ) {
ReadWaveFormsHeaders ( f , wave_forms , session , usage ) ;
initialized = true ;
}
f . seek ( kMainHeaderSize + wave_forms . size ( ) * kDescriptionHeaderSize ) ;
2023-05-13 21:23:22 +00:00
QVector < qint16 > chunk ( std : : max_element ( wave_forms . cbegin ( ) , wave_forms . cend ( ) , [ ] ( const ChunkData & lhs , const ChunkData & rhs ) {
return lhs . samples_by_chunk < rhs . samples_by_chunk ;
} ) - > samples_by_chunk ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
while ( ! f . atEnd ( ) ) {
2023-05-13 21:23:22 +00:00
for ( int i = 0 ; i < wave_forms . size ( ) ; i + + ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto & wave_form = wave_forms [ i ] . event_list ;
const auto samples_by_chunk_actual = wave_forms [ i ] . samples_by_chunk ;
auto & start_time_current = wave_forms [ i ] . start_time ;
auto & total_samples_by_chunk = wave_forms [ i ] . total_samples_by_chunk ;
const auto sample_rate = wave_forms [ i ] . sample_rate ;
const auto duration = samples_by_chunk_actual * 1000.0 / sample_rate ;
const auto readed = f . read ( reinterpret_cast < char * > ( chunk . data ( ) ) , chunk . size ( ) * sizeof ( qint16 ) ) ;
if ( wave_form ) {
const auto readed_elements = readed / sizeof ( qint16 ) ;
if ( readed_elements ! = samples_by_chunk_actual ) {
std : : fill ( std : : begin ( chunk ) + readed_elements , std : : end ( chunk ) , 0 ) ;
}
wave_form - > AddWaveform ( start_time_current + kDateTimeOffset , chunk . data ( ) , samples_by_chunk_actual , duration ) ;
}
start_time_current + = duration ;
total_samples_by_chunk + = samples_by_chunk_actual ;
}
}
} ) ;
2023-05-13 21:23:22 +00:00
QVector < qint16 > chunk ;
for ( int i = 0 ; i < wave_forms . size ( ) ; i + + ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto & wave_form = wave_forms [ i ] ;
const auto expected_samples = usage . start_time . msecsTo ( usage . end_time ) / 1000.0 * wave_form . sample_rate ;
if ( wave_form . total_samples_by_chunk < expected_samples ) {
chunk . resize ( expected_samples - wave_form . total_samples_by_chunk ) ;
if ( wave_form . event_list ) {
const auto duration = chunk . size ( ) * 1000.0 / wave_form . sample_rate ;
wave_form . event_list - > AddWaveform ( wave_form . start_time + kDateTimeOffset , chunk . data ( ) , chunk . size ( ) , duration ) ;
}
}
}
}
void LoadStats ( const UsageData & /*usage_data*/ , Session * session ) {
// session->settings[CPAP_AHI] = usage_data.countAHI;
// session->setCount(CPAP_AI, usage_data.countAI);
// session->setCount(CPAP_CAI, usage_data.countCAI);
// session->setCount(CPAP_HI, usage_data.countHI);
// session->setCount(CPAP_Obstructive, usage_data.countOAI);
// session->settings[CPAP_RERA] = usage_data.countRERA;
// session->settings[CPAP_Snore] = usage_data.countSNI;
session - > settings [ CPAP_Mode ] = MODE_APAP ;
}
UsageData ReadUsage ( const QString & session_folder_path , const QString & usage_number ) {
UsageData usage_data ;
usage_data . number = usage_number ;
const auto session_stat_path = session_folder_path + QDir : : separator ( ) + " STAT " + usage_number ;
if ( ! QFile : : exists ( session_stat_path ) ) {
qDebug ( ) < < " Resvent Data card has no " < < session_stat_path ;
return usage_data ;
}
QFile f ( session_stat_path ) ;
f . open ( QIODevice : : ReadOnly | QIODevice : : Text ) ;
f . seek ( 4 ) ;
while ( ! f . atEnd ( ) ) {
QString line = f . readLine ( ) . trimmed ( ) ;
const auto elems = line . split ( " = " ) ;
2023-05-10 19:41:27 +00:00
Q_ASSERT ( elems . size ( ) = = 2 ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
if ( elems [ 0 ] = = " secStart " ) {
usage_data . start_time = QDateTime : : fromTime_t ( std : : stoi ( elems [ 1 ] . toStdString ( ) ) ) ;
}
else if ( elems [ 0 ] = = " secUsed " ) {
usage_data . end_time = QDateTime : : fromTime_t ( usage_data . start_time . toTime_t ( ) + std : : stoi ( elems [ 1 ] . toStdString ( ) ) ) ;
}
else if ( elems [ 0 ] = = " cntAHI " ) {
usage_data . countAHI = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntOAI " ) {
usage_data . countOAI = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntCAI " ) {
usage_data . countCAI = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntAI " ) {
usage_data . countAI = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntHI " ) {
usage_data . countHI = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntRERA " ) {
usage_data . countRERA = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntSNI " ) {
usage_data . countSNI = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
else if ( elems [ 0 ] = = " cntBreath " ) {
usage_data . countBreath = std : : stoi ( elems [ 1 ] . toStdString ( ) ) ;
}
}
return usage_data ;
}
2023-05-13 21:23:22 +00:00
QVector < UsageData > GetDifferentUsage ( const QString & session_folder_path ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
QDir session_folder ( session_folder_path ) ;
const auto stat_files = session_folder . entryList ( QStringList ( ) < < " STAT* " , QDir : : Files , QDir : : Name ) ;
2023-05-13 21:23:22 +00:00
QVector < UsageData > usage_data ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
std : : for_each ( stat_files . cbegin ( ) , stat_files . cend ( ) , [ & ] ( const QString & stat_file ) {
if ( stat_file . size ( ) ! = 6 ) {
return ;
}
auto usageData = ReadUsage ( session_folder_path , stat_file . right ( 2 ) ) ;
usage_data . push_back ( usageData ) ;
} ) ;
return usage_data ;
}
2023-10-30 23:20:20 +00:00
int ResventLoader : : LoadSession ( const QString & dirpath , const QDate & session_date , Machine * machine ) {
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
const auto session_folder_path = GetSessionFolder ( dirpath , session_date ) ;
const auto different_usage = GetDifferentUsage ( session_folder_path ) ;
2023-11-11 18:10:48 +00:00
//return std::accumulate(different_usage.cbegin(), different_usage.cend(), 0, [&](int base, const UsageData& usage)
// std::accumulate(different_usage.cbegin(), different_usage.cend(), 0, [&](int base, const UsageData& usage)
int base = 0 ;
for ( auto usage : different_usage )
{
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
if ( machine - > SessionExists ( usage . start_time . toMSecsSinceEpoch ( ) + kDateTimeOffset ) ) {
2023-11-11 18:10:48 +00:00
// session alreadt exists
2023-11-12 00:30:07 +00:00
//return base;
continue ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
Session * session = new Session ( machine , usage . start_time . toMSecsSinceEpoch ( ) + kDateTimeOffset ) ;
session - > SetChanged ( true ) ;
session - > really_set_first ( usage . start_time . toMSecsSinceEpoch ( ) + kDateTimeOffset ) ;
session - > really_set_last ( usage . end_time . toMSecsSinceEpoch ( ) + kDateTimeOffset ) ;
LoadStats ( usage , session ) ;
LoadWaveForms ( session_folder_path , session , usage ) ;
LoadOtherWaveForms ( session_folder_path , session , usage ) ;
LoadEvents ( session_folder_path , session , usage ) ;
session - > UpdateSummaries ( ) ;
session - > Store ( machine - > getDataPath ( ) ) ;
machine - > AddSession ( session ) ;
2023-10-30 23:20:20 +00:00
emit setProgressValue ( + + progress ) ;
2023-10-30 16:49:13 +00:00
QCoreApplication : : processEvents ( ) ;
2023-11-11 18:10:48 +00:00
+ + base ;
} ;
return base ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
}
///////////////////////////////////////////////////////////////////////////////////////////
// Sorted EDF files that need processing into date records according to ResMed noon split
///////////////////////////////////////////////////////////////////////////////////////////
int ResventLoader : : Open ( const QString & dirpath )
{
const auto machine_info = PeekInfo ( dirpath ) ;
// Abort if no serial number
if ( machine_info . serial . isEmpty ( ) ) {
qDebug ( ) < < " Resvent Data card has no valid serial number in " < < kResventSysConfigFilename ;
return - 1 ;
}
const auto sessions_date = GetSessionsDate ( dirpath ) ;
Machine * machine = p_profile - > CreateMachine ( machine_info ) ;
int new_sessions = 0 ;
std : : for_each ( sessions_date . cbegin ( ) , sessions_date . cend ( ) , [ & ] ( const QDate & session_date ) {
new_sessions + = LoadSession ( dirpath , session_date , machine ) ;
} ) ;
machine - > Save ( ) ;
return new_sessions ;
}
void ResventLoader : : initChannels ( )
{
}
ChannelID ResventLoader : : PresReliefMode ( ) { return 0 ; }
ChannelID ResventLoader : : PresReliefLevel ( ) { return 0 ; }
ChannelID ResventLoader : : CPAPModeChannel ( ) { return 0 ; }
bool resvent_initialized = false ;
void ResventLoader : : Register ( )
{
if ( resvent_initialized ) { return ; }
qDebug ( ) < < " Registering ResventLoader " ;
RegisterLoader ( new ResventLoader ( ) ) ;
resvent_initialized = true ;
}