2020-01-23 17:51:58 +00:00
/* SleepLib Viatom Loader Implementation
*
2024-01-13 20:27:48 +00:00
* Copyright ( c ) 2020 - 2024 The OSCAR Team
2020-01-24 00:25:06 +00:00
* ( Initial importer written by dave madden < dhm @ mersenne . com > )
2020-01-23 17:51:58 +00:00
*
* 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 . */
//********************************************************************************************
2021-03-22 22:43:14 +00:00
// Please only INCREMENT the viatom_data_version in viatom_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
2020-01-23 17:51:58 +00:00
//********************************************************************************************
# include <QDir>
# include <QTextStream>
2020-01-25 01:10:41 +00:00
# include <QApplication>
# include <QMessageBox>
2020-01-23 17:51:58 +00:00
# include "viatom_loader.h"
# include "SleepLib/machine.h"
2021-09-20 17:37:03 +00:00
// TODO: Merge this with PRS1 macros and generalize for all loaders.
# define SESSIONID m_session->session()
# define UNEXPECTED_VALUE(SRC, VALS) { \
QString message = QString ( " %1:%2: %3 = %4 != %5 " ) . arg ( __func__ ) . arg ( __LINE__ ) . arg ( # SRC ) . arg ( SRC ) . arg ( VALS ) ; \
qWarning ( ) < < SESSIONID < < message ; \
s_unexpectedMessages + = message ; \
}
# define CHECK_VALUE(SRC, VAL) if ((SRC) != (VAL)) UNEXPECTED_VALUE(SRC, VAL)
# define CHECK_VALUES(SRC, VAL1, VAL2) if ((SRC) != (VAL1) && (SRC) != (VAL2)) UNEXPECTED_VALUE(SRC, #VAL1 " or " #VAL2)
// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails
2020-01-25 01:10:41 +00:00
static QSet < QString > s_unexpectedMessages ;
2020-01-23 17:51:58 +00:00
bool
ViatomLoader : : Detect ( const QString & path )
{
2022-02-27 16:50:10 +00:00
// This is only used for CPAP devices, when detecting CPAP cards.
2020-01-24 00:25:06 +00:00
qDebug ( ) < < " ViatomLoader::Detect( " < < path < < " ) " ;
2020-01-26 23:28:10 +00:00
return false ;
2020-01-23 17:51:58 +00:00
}
int
2021-04-24 07:44:27 +00:00
ViatomLoader : : Open ( const QStringList & paths )
2020-01-23 17:51:58 +00:00
{
2021-04-24 07:44:27 +00:00
qDebug ( ) < < " ViatomLoader::Open( " < < paths . join ( " ; " ) < < " ) " ;
2020-04-26 18:06:48 +00:00
m_mach = nullptr ;
2020-01-26 23:13:09 +00:00
int imported = 0 ;
int found = 0 ;
2020-01-26 23:28:10 +00:00
s_unexpectedMessages . clear ( ) ;
2021-02-22 20:30:04 +00:00
2021-04-24 07:44:27 +00:00
int size = paths . size ( ) ;
for ( int i = 0 ; i < size ; i + + ) {
if ( isAborted ( ) ) {
break ;
2020-01-26 23:13:09 +00:00
}
// This filename has already been filtered by QFileDialog.
2021-04-24 07:44:27 +00:00
int ok = OpenFile ( paths [ i ] ) ;
if ( ok > 0 ) {
2020-04-26 18:06:48 +00:00
imported + + ;
2021-04-24 07:44:27 +00:00
} else if ( ok < 0 ) {
// Stop on error...
break ;
2020-04-26 18:06:48 +00:00
}
2020-01-26 23:13:09 +00:00
found + + ;
2021-04-24 07:44:27 +00:00
emit setProgressValue ( i + 1 ) ;
QCoreApplication : : processEvents ( ) ;
2020-01-26 23:13:09 +00:00
}
if ( ! found ) {
return - 1 ;
}
2020-04-26 18:06:48 +00:00
Machine * mach = m_mach ;
2022-02-27 16:50:10 +00:00
if ( imported & & mach = = nullptr ) qWarning ( ) < < " No device record created? " ;
2020-01-27 00:50:18 +00:00
if ( mach ) {
qDebug ( ) < < " Imported " < < imported < < " sessions " ;
mach - > Save ( ) ;
mach - > SaveSummaryCache ( ) ;
p_profile - > StoreMachines ( ) ;
}
2020-01-26 23:28:10 +00:00
if ( mach & & s_unexpectedMessages . count ( ) > 0 & & p_profile - > session - > warnOnUnexpectedData ( ) ) {
2022-02-27 16:50:10 +00:00
// Compare this to the list of messages previously seen for this device
2020-01-26 23:28:10 +00:00
// and only alert if there are new ones.
QSet < QString > newMessages = s_unexpectedMessages - mach - > previouslySeenUnexpectedData ( ) ;
if ( newMessages . count ( ) > 0 ) {
// TODO: Rework the importer call structure so that this can become an
// emit statement to the appropriate import job.
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
QObject : : tr ( " Untested Data " ) ,
QObject : : tr ( " Your Viatom device generated data that OSCAR has never seen before. " ) + " \n \n " +
2020-01-27 14:15:15 +00:00
QObject : : tr ( " The imported data may not be entirely accurate, so the developers would like a copy of your Viatom files to make sure OSCAR is handling the data correctly. " )
2020-01-26 23:28:10 +00:00
, QMessageBox : : Ok ) ;
mach - > previouslySeenUnexpectedData ( ) + = newMessages ;
}
}
2021-04-24 07:44:27 +00:00
return found ;
2020-01-23 17:51:58 +00:00
}
2021-04-24 07:44:27 +00:00
int ViatomLoader : : OpenFile ( const QString & filename )
2020-01-23 17:51:58 +00:00
{
2020-01-26 23:28:10 +00:00
Machine * mach = nullptr ;
2021-04-24 07:44:27 +00:00
bool existing = false ;
2021-02-22 20:30:04 +00:00
2021-04-24 07:44:27 +00:00
Session * sess = ParseFile ( filename , & existing ) ;
2020-01-26 23:13:09 +00:00
if ( sess ) {
2020-01-24 00:11:05 +00:00
SaveSessionToDatabase ( sess ) ;
2020-01-26 23:28:10 +00:00
mach = sess - > machine ( ) ;
2020-04-26 18:06:48 +00:00
m_mach = mach ;
2021-04-24 07:44:27 +00:00
return 1 ;
2020-01-24 00:11:05 +00:00
}
2020-01-25 01:10:41 +00:00
2021-04-24 07:44:27 +00:00
return existing ? 0 : - 1 ; // -1 = error
2020-01-24 00:11:05 +00:00
}
2021-04-24 07:44:27 +00:00
Session * ViatomLoader : : ParseFile ( const QString & filename , bool * existing )
2020-01-24 00:11:05 +00:00
{
2021-04-24 07:44:27 +00:00
if ( existing ) {
* existing = false ;
}
2020-01-24 00:25:06 +00:00
QFile file ( filename ) ;
if ( ! file . open ( QFile : : ReadOnly ) ) {
qDebug ( ) < < " Couldn't open Viatom data file " < < filename ;
return nullptr ;
2020-01-23 17:51:58 +00:00
}
2024-08-18 15:32:22 +00:00
// Read in Viatom database version number
QByteArray data = file . read ( 2 ) ;
if ( data . size ( ) < 2 ) {
qDebug ( ) < < filename < < " too short for a Viatom data file " ;
return nullptr ;
}
const unsigned char * header = ( const unsigned char * ) data . constData ( ) ;
int sig = header [ 0 ] | ( header [ 1 ] < < 8 ) ;
std : : unique_ptr < ViatomFile > v = nullptr ;
switch ( sig ) {
case 0x0003 :
case 0x0005 :
v = std : : unique_ptr < ViatomFile > ( new ViatomFile ( file ) ) ;
break ;
case 0x0301 :
v = std : : unique_ptr < O2RingS > ( new O2RingS ( file ) ) ;
break ;
default :
qDebug ( ) < < filename < < " Unrecognized DB version number in Viatom data file " < < sig ;
return nullptr ;
}
if ( ! file . seek ( 0 ) ) {
qDebug ( ) < < filename < < " unable to seek to begining of file " ;
return nullptr ;
}
// Parse header specific to database version number
if ( v - > ParseHeader ( ) = = false ) {
2020-01-24 00:25:06 +00:00
return nullptr ;
}
2020-01-27 17:26:35 +00:00
MachineInfo info = newInfo ( ) ;
// Check whether the enclosing folder looks like a Viatom serial number, and if so, use it.
QString foldername = QFileInfo ( filename ) . dir ( ) . dirName ( ) ;
if ( foldername . length ( ) > = 9 ) {
bool numeric ;
2021-02-22 20:30:04 +00:00
foldername . rightRef ( 4 ) . toInt ( & numeric ) ;
2020-01-27 17:26:35 +00:00
if ( numeric ) {
info . serial = foldername ;
}
}
Machine * mach = p_profile - > CreateMachine ( info ) ;
2020-01-24 00:25:06 +00:00
2024-08-18 15:32:22 +00:00
if ( mach - > SessionExists ( v - > sessionid ( ) ) ) {
2020-01-26 22:20:58 +00:00
// Skip already imported session
2020-01-27 00:50:18 +00:00
//qDebug() << filename << "session already exists, skipping" << v.sessionid();
2021-04-24 07:44:27 +00:00
if ( existing ) {
// Inform the caller (if they are interested) that this session was already imported
* existing = true ;
}
2020-01-26 22:20:58 +00:00
return nullptr ;
2020-01-24 00:25:06 +00:00
}
2020-01-26 22:20:58 +00:00
2024-08-18 15:32:22 +00:00
qint64 time_ms = v - > timestamp ( ) ;
m_session = new Session ( mach , v - > sessionid ( ) ) ;
2020-01-26 22:20:58 +00:00
m_session - > set_first ( time_ms ) ;
2020-01-24 00:25:06 +00:00
2024-08-18 15:32:22 +00:00
QList < ViatomFile : : Record > records = v - > ReadData ( ) ;
m_step = v - > duration ( ) / records . size ( ) * 1000L ;
2020-01-23 17:51:58 +00:00
2020-01-24 20:20:00 +00:00
// Import data
for ( auto & rec : records ) {
2020-01-25 21:55:32 +00:00
if ( rec . oximetry_invalid ) {
EndEventList ( OXI_Pulse , time_ms ) ;
EndEventList ( OXI_SPO2 , time_ms ) ;
} else {
2021-09-20 17:37:03 +00:00
// Viatom advertises a range of 30 - 250 bpm.
if ( rec . hr < 30 | | rec . hr > 250 ) {
UNEXPECTED_VALUE ( rec . hr , " 30-250 " ) ;
}
2020-01-25 21:55:32 +00:00
AddEvent ( OXI_Pulse , time_ms , rec . hr ) ;
2021-09-20 17:37:03 +00:00
if ( rec . spo2 = = 0xFF ) {
// When the readings fall below 61%, Viatom devices record 0xFF for SpO2.
// The official software discards these readings.
// TODO: Consider whether to import these as 60% since they reflect hypoxia.
EndEventList ( OXI_SPO2 , time_ms ) ;
//qDebug() << "<61% at" << QDateTime::fromMSecsSinceEpoch(time_ms);
} else {
// Viatom advertises (and graphs) a range of 70% - 99%, but apparently records down to 61%.
// The official software graphs 61%-70% as 70%.
// TODO: Consider whether we should import 61%-70% as 70% to match the official reports.
if ( rec . spo2 < 61 | | rec . spo2 > 99 ) {
UNEXPECTED_VALUE ( rec . spo2 , " 61-99% " ) ;
}
AddEvent ( OXI_SPO2 , time_ms , rec . spo2 ) ;
}
2020-01-25 21:55:32 +00:00
}
2020-01-27 00:50:18 +00:00
AddEvent ( POS_Movement , time_ms , rec . motion ) ;
2020-01-25 21:55:32 +00:00
time_ms + = m_step ;
2020-01-24 20:20:00 +00:00
}
2020-01-25 21:55:32 +00:00
EndEventList ( OXI_Pulse , time_ms ) ;
EndEventList ( OXI_SPO2 , time_ms ) ;
2020-01-27 00:50:18 +00:00
EndEventList ( POS_Movement , time_ms ) ;
2020-01-25 21:55:32 +00:00
m_session - > set_last ( time_ms ) ;
2020-01-23 17:51:58 +00:00
2020-01-25 21:55:32 +00:00
return m_session ;
2020-01-24 00:11:05 +00:00
}
void ViatomLoader : : SaveSessionToDatabase ( Session * sess )
{
Machine * mach = sess - > machine ( ) ;
2021-02-22 20:30:04 +00:00
2020-01-24 00:25:06 +00:00
sess - > SetChanged ( true ) ;
mach - > AddSession ( sess ) ;
2020-01-23 17:51:58 +00:00
}
2020-01-25 21:55:32 +00:00
void ViatomLoader : : AddEvent ( ChannelID channel , qint64 t , EventDataType value )
{
EventList * C = m_importChannels [ channel ] ;
if ( C = = nullptr ) {
C = m_session - > AddEventList ( channel , EVL_Waveform , 1.0 , 0.0 , 0.0 , 0.0 , m_step ) ;
Q_ASSERT ( C ) ; // Once upon a time AddEventList could return nullptr, but not any more.
m_importChannels [ channel ] = C ;
}
// Add the event
C - > AddEvent ( t , value ) ;
m_importLastValue [ channel ] = value ;
}
void ViatomLoader : : EndEventList ( ChannelID channel , qint64 /*t*/ )
{
EventList * C = m_importChannels [ channel ] ;
if ( C ! = nullptr ) {
// The below would be needed for square charts if the first sample represents
// the 4 seconds following the starting timestamp:
//C->AddEvent(t, m_importLastValue[channel]);
2021-02-22 20:30:04 +00:00
2020-01-25 21:55:32 +00:00
// Mark this channel's event list as ended.
m_importChannels [ channel ] = nullptr ;
}
}
2020-01-26 22:51:56 +00:00
QStringList ViatomLoader : : getNameFilter ( )
{
2021-09-15 20:06:21 +00:00
// Sometimes the files have a SleepU_ or O2Ring_ prefix.
2021-09-20 18:08:49 +00:00
// Sometimes they have punctuation in the timestamp.
// Note that ":" is not allowed on macOS, so Mac users will need to rename their files in order to select and import them.
2023-01-15 19:50:02 +00:00
return QStringList ( { " *20[0-5][0-9][01][0-9][0-3][0-9][012][0-9][0-5][0-9][0-5][0-9]* " ,
" *20[0-5][0-9]-[01][0-9]-[0-3][0-9] [012][0-9]:[0-5][0-9]:[0-5][0-9]* "
2021-09-20 18:08:49 +00:00
} ) ;
2020-01-26 22:51:56 +00:00
}
2020-01-23 17:51:58 +00:00
static bool viatom_initialized = false ;
void
2020-01-24 00:25:06 +00:00
ViatomLoader : : Register ( )
2020-01-23 17:51:58 +00:00
{
if ( ! viatom_initialized ) {
2020-01-24 00:25:06 +00:00
qDebug ( " Registering ViatomLoader " ) ;
RegisterLoader ( new ViatomLoader ( ) ) ;
//InitModelMap();
viatom_initialized = true ;
}
2020-01-23 17:51:58 +00:00
}
2020-01-24 20:20:00 +00:00
// ===============================================================================================
2021-09-20 17:37:03 +00:00
# undef SESSIONID
# define SESSIONID this->m_sessionid
2021-02-22 20:30:04 +00:00
2020-01-25 01:10:41 +00:00
ViatomFile : : ViatomFile ( QFile & file ) : m_file ( file )
{
}
2020-01-24 20:20:00 +00:00
2024-08-18 15:32:22 +00:00
QDateTime ViatomFile : : getFilenameTimestamp ( )
{
QString date_string = QFileInfo ( m_file ) . fileName ( ) . section ( " _ " , - 1 ) ; // Strip any SleepU_ etc. prefix.
int lastPoint = date_string . lastIndexOf ( " . " ) ; // Added to strip off any filename extension
date_string = date_string . left ( lastPoint ) ;
QString format_string = " yyyyMMddHHmmss " ;
if ( date_string . contains ( " : " ) ) {
format_string = " yyyy-MM-dd HH:mm:ss " ;
}
return QDateTime : : fromString ( date_string , format_string ) ;
}
2020-01-25 01:10:41 +00:00
bool ViatomFile : : ParseHeader ( )
{
static const int HEADER_SIZE = 40 ;
QByteArray data = m_file . read ( HEADER_SIZE ) ;
if ( data . size ( ) < HEADER_SIZE ) {
qDebug ( ) < < m_file . fileName ( ) < < " too short for a Viatom data file " ;
return false ;
}
2020-01-24 20:20:00 +00:00
2020-01-25 01:10:41 +00:00
const unsigned char * header = ( const unsigned char * ) data . constData ( ) ;
int sig = header [ 0 ] | ( header [ 1 ] < < 8 ) ;
int year = header [ 2 ] | ( header [ 3 ] < < 8 ) ;
int month = header [ 4 ] ;
int day = header [ 5 ] ;
int hour = header [ 6 ] ;
int min = header [ 7 ] ;
int sec = header [ 8 ] ;
2023-12-03 15:30:17 +00:00
switch ( sig ) { //Viatom database version number - Crimson Nape
2021-02-22 20:30:04 +00:00
case 0x0003 :
2023-12-03 15:30:17 +00:00
case 0x0005 :
2021-02-22 20:30:04 +00:00
break ;
default :
2023-12-03 15:30:17 +00:00
qDebug ( ) < < m_file . fileName ( ) < < " Unrecognized DB version number in Viatom data file " < < sig ;
2021-02-22 20:30:04 +00:00
return false ;
break ;
}
2022-01-11 01:14:04 +00:00
m_sig = sig ;
CHECK_VALUES ( m_sig , 3 , 5 ) ;
2021-02-22 20:30:04 +00:00
2020-01-25 01:10:41 +00:00
if ( ( year < 2015 | | year > 2059 ) | | ( month < 1 | | month > 12 ) | | ( day < 1 | | day > 31 ) | |
( hour > 23 ) | | ( min > 59 ) | | ( sec > 59 ) ) {
qDebug ( ) < < m_file . fileName ( ) < < " invalid timestamp in Viatom data file " ;
2020-01-24 20:20:00 +00:00
return false ;
}
2020-01-25 21:55:32 +00:00
// It's unclear what the starting timestamp represents: is it the time at which
// the device starts measuring data, and the first sample is 4s after that? Or
// is the starting timestamp the time at which the first 4s average is reported
// (and the first 4 seconds being average precede the starting timestamp)?
//
// If the former, then the chart draws the first sample too early (right at the
// starting timestamp). Technically these should probably be square charts, but
// the code currently forces them to be non-square.
2020-01-25 01:10:41 +00:00
QDateTime data_timestamp = QDateTime ( QDate ( year , month , day ) , QTime ( hour , min , sec ) ) ;
2024-08-18 15:32:22 +00:00
QDateTime filename_timestamp = getFilenameTimestamp ( ) ;
2021-09-17 13:06:37 +00:00
if ( filename_timestamp . isValid ( ) ) {
if ( filename_timestamp ! = data_timestamp ) {
// TODO: Once there's a better/easier way to adjust session times within OSCAR, we can remove the below.
qDebug ( ) < < m_file . fileName ( ) < < " Using filename timestamp " < < filename_timestamp . toString ( " yyyy-MM-dd HH:mm:ss " )
< < " instead of header timestamp " < < data_timestamp . toString ( " yyyy-MM-dd HH:mm:ss " ) ;
data_timestamp = filename_timestamp ;
}
} else {
qWarning ( ) < < m_file . fileName ( ) < < " invalid timestamp in Viatom filename " ;
}
2020-01-24 20:20:00 +00:00
m_timestamp = data_timestamp . toMSecsSinceEpoch ( ) ;
2020-01-25 01:10:41 +00:00
m_sessionid = m_timestamp / 1000L ;
2022-01-11 01:14:04 +00:00
int filesize = header [ 9 ] | ( header [ 10 ] < < 8 ) | ( header [ 11 ] < < 16 ) ; // possibly 32-bit
2020-01-25 01:10:41 +00:00
CHECK_VALUE ( header [ 12 ] , 0 ) ;
m_duration = header [ 13 ] | ( header [ 14 ] < < 8 ) ; // possibly 32-bit
CHECK_VALUE ( header [ 15 ] , 0 ) ;
CHECK_VALUE ( header [ 16 ] , 0 ) ;
//int spo2_avg = header[17];
//int spo2_min = header[18];
//int spo2_3pct = header[19]; // number of events
2020-05-06 15:42:03 +00:00
//int spo2_4pct = header[20]; // number of events
//CHECK_VALUE(header[21], 0); // ??? sometimes nonzero; maybe pulse spike, not a threshold of SpO2 or pulse, not always smaller than spo2_4pct
2020-02-07 18:54:28 +00:00
//int time_under_90pct = header[22] | (header[23] << 8); // in seconds
2020-01-25 01:10:41 +00:00
//int events_under_90pct = header[24]; // number of distinct events
//float o2_score = header[25] * 0.1;
2021-09-26 16:25:49 +00:00
//CHECK_VALUES(header[26], 0, 4); // number of steps taken (when nonzero, only reported by some models)
2020-01-25 01:10:41 +00:00
CHECK_VALUE ( header [ 27 ] , 0 ) ;
CHECK_VALUE ( header [ 28 ] , 0 ) ;
CHECK_VALUE ( header [ 29 ] , 0 ) ;
2021-09-20 17:37:03 +00:00
//CHECK_VALUE(header[30], 0); // average pulse rate (when nonzero)
2020-01-25 01:10:41 +00:00
CHECK_VALUE ( header [ 31 ] , 0 ) ;
CHECK_VALUE ( header [ 32 ] , 0 ) ;
CHECK_VALUE ( header [ 33 ] , 0 ) ;
CHECK_VALUE ( header [ 34 ] , 0 ) ;
CHECK_VALUE ( header [ 35 ] , 0 ) ;
CHECK_VALUE ( header [ 36 ] , 0 ) ;
CHECK_VALUE ( header [ 37 ] , 0 ) ;
CHECK_VALUE ( header [ 38 ] , 0 ) ;
CHECK_VALUE ( header [ 39 ] , 0 ) ;
// Calculate timing resolution (in ms) of the data
qint64 datasize = m_file . size ( ) - HEADER_SIZE ;
m_record_count = datasize / RECORD_SIZE ;
m_resolution = m_duration / m_record_count * 1000L ;
2022-01-11 01:14:04 +00:00
if ( m_resolution = = 2000 & & m_sig = = 3 ) {
2020-01-25 01:10:41 +00:00
// Interestingly the file size in the header corresponds the number of
// distinct samples. These files actually double-report each sample!
// So this resolution isn't really the real one. The importer should
// calculate resolution from duration / record count after reading the
// records, which will be deduplicated.
CHECK_VALUE ( filesize , ( ( m_file . size ( ) - HEADER_SIZE ) / 2 ) + HEADER_SIZE ) ;
} else {
CHECK_VALUE ( filesize , m_file . size ( ) ) ;
}
CHECK_VALUES ( m_resolution , 2000 , 4000 ) ;
2021-09-15 21:02:58 +00:00
if ( true ) { // TODO: We need CheckMe sample data where this doesn't hold true.
CHECK_VALUE ( datasize % RECORD_SIZE , 0 ) ;
CHECK_VALUE ( m_duration % m_record_count , 0 ) ;
}
2020-01-24 20:20:00 +00:00
2020-01-26 22:20:58 +00:00
//qDebug().noquote() << m_file.fileName() << ts(m_timestamp) << dur(m_duration * 1000L) << ":" << m_record_count << "records @" << m_resolution << "ms";
2020-01-24 20:20:00 +00:00
return true ;
}
QList < ViatomFile : : Record > ViatomFile : : ReadData ( )
{
QByteArray data = m_file . readAll ( ) ;
QDataStream in ( data ) ;
in . setByteOrder ( QDataStream : : LittleEndian ) ;
2021-09-15 21:02:58 +00:00
2020-01-24 20:20:00 +00:00
QList < ViatomFile : : Record > records ;
2021-09-15 21:02:58 +00:00
2020-01-24 20:20:00 +00:00
// Read all Pulse, SPO2 and Motion data
do {
ViatomFile : : Record rec ;
2020-02-03 20:37:37 +00:00
in > > rec . spo2 > > rec . hr > > rec . oximetry_invalid > > rec . motion > > rec . vibration ;
2021-09-15 21:02:58 +00:00
CHECK_VALUES ( rec . oximetry_invalid , 0 , 0xFF ) ;
if ( rec . vibration ) {
CHECK_VALUES ( rec . vibration , 0x40 , 0x80 ) ; // 0x40 or 0x80 when vibration is triggered
}
// Invalid readings indicate any interruption in the measurements, whether
// transitory (e.g. due to movement) or when the device is removed at the end of a session.
2020-01-25 21:55:32 +00:00
if ( rec . oximetry_invalid = = 0xFF ) {
CHECK_VALUE ( rec . spo2 , 0xFF ) ;
2021-09-15 21:02:58 +00:00
CHECK_VALUE ( rec . hr , 0xFF ) ;
2020-01-25 21:55:32 +00:00
}
2020-01-24 20:20:00 +00:00
records . append ( rec ) ;
2021-09-15 21:02:58 +00:00
} while ( records . size ( ) < m_record_count ) ;
2020-01-24 20:20:00 +00:00
2022-01-11 01:14:04 +00:00
// It turns out 2s files are actually just double-reported samples on older models!
2020-01-25 01:10:41 +00:00
if ( m_resolution = = 2000 ) {
2020-01-25 19:50:45 +00:00
QList < ViatomFile : : Record > dedup ;
bool all_are_duplicated = true ;
2022-05-09 14:51:17 +00:00
if ( ( records . size ( ) % 2 ) ! = 0 ) {
// An odd number of samples inherently can't be all duplicates.
all_are_duplicated = false ;
} else {
for ( int i = 0 ; i < records . size ( ) ; i + = 2 ) {
auto & a = records . at ( i ) ;
auto & b = records . at ( i + 1 ) ;
if ( a . spo2 ! = b . spo2
| | a . hr ! = b . hr
| | a . oximetry_invalid ! = b . oximetry_invalid
| | a . motion ! = b . motion
| | a . vibration ! = b . vibration ) {
all_are_duplicated = false ;
break ;
}
dedup . append ( a ) ;
2020-01-25 19:50:45 +00:00
}
}
2022-01-11 01:14:04 +00:00
if ( m_sig = = 5 ) {
// Confirm that CheckMe O2 Max is a true 2s sample rate.
CHECK_VALUE ( all_are_duplicated , false ) ;
} else {
// Confirm that older models are actually a 4s sample rate.
CHECK_VALUE ( m_sig , 3 ) ;
CHECK_VALUE ( all_are_duplicated , true ) ;
}
2020-01-25 19:50:45 +00:00
if ( all_are_duplicated ) {
// Return the deduplicated list.
records = dedup ;
2020-01-25 01:10:41 +00:00
}
}
2022-01-11 01:14:04 +00:00
if ( m_sig = = 5 ) {
CHECK_VALUES ( duration ( ) / records . size ( ) , 2 , 4 ) ; // We've seen 2s and 4s resolution.
} else {
CHECK_VALUE ( duration ( ) / records . size ( ) , 4 ) ; // We've only seen 4s true resolution so far.
}
2020-01-25 01:10:41 +00:00
2020-01-24 20:20:00 +00:00
return records ;
}
2020-01-25 19:50:45 +00:00
2024-08-18 15:32:22 +00:00
O2RingS : : O2RingS ( QFile & file ) : ViatomFile ( file )
{
}
bool O2RingS : : ParseHeader ( )
{
// For the O2Ring S, the header only contains the signature
// Additional metadata is stored at the end of the file
// The record count is stored 36 bytes prior to EOF
int record_count_loc = m_file . size ( ) - 36 ;
if ( record_count_loc < 0 | | ! m_file . seek ( record_count_loc ) ) {
qDebug ( ) < < m_file . fileName ( ) < < " error locating Viatom record count " ;
return false ;
}
// read record count as a 2-byte little endian value
// max number of records in a O2Ring S file is 36000
QDataStream in ( m_file . read ( 2 ) ) ;
in . setByteOrder ( QDataStream : : LittleEndian ) ;
quint16 record_count ;
in > > record_count ;
m_sig = 0x0301 ;
m_record_count = m_duration = record_count ;
m_timestamp = getFilenameTimestamp ( ) . toMSecsSinceEpoch ( ) ;
m_sessionid = m_timestamp / 1000L ;
m_resolution = 1000 ;
// advance past the header
return m_file . seek ( 10 ) ;
}
QList < ViatomFile : : Record > O2RingS : : ReadData ( )
{
QList < ViatomFile : : Record > records ;
// Read all Pulse, SPO2 and Motion data
// 0xFF for spo2 or hr indicates an interruption in measurement
// Vibration data is likely stored in a variable length block following the
// fixed-width pulse/SPO2/motion data. Zero out for now since OSCAR doesn't
// use this data.
QDataStream in ( m_file . readAll ( ) ) ;
do {
ViatomFile : : Record rec ;
in > > rec . spo2 > > rec . hr > > rec . motion ;
rec . oximetry_invalid = ( rec . spo2 = = 0xFF | | rec . hr = = 0xFF ) ? 0xFF : 0 ;
rec . vibration = 0 ;
records . append ( rec ) ;
} while ( records . size ( ) < m_record_count ) ;
// Confirm that we have a 1s sample rate
CHECK_VALUE ( duration ( ) / records . size ( ) , 1 ) ;
return records ;
}