mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-04 02:00:43 +00:00
176 lines
6.2 KiB
C++
176 lines
6.2 KiB
C++
/* EDF Parser Header
|
|
*
|
|
* Copyright (c) 2019 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. */
|
|
|
|
#ifndef EDFPARSER_H
|
|
#define EDFPARSER_H
|
|
|
|
#include <QString>
|
|
#include <QVector>
|
|
#include <QHash>
|
|
#include <QList>
|
|
#include <QDateTime>
|
|
|
|
// #include "SleepLib/common.h"
|
|
|
|
typedef float EventDataType;
|
|
|
|
const QString STR_ext_EDF = "edf";
|
|
const QString STR_ext_gz = ".gz";
|
|
|
|
const char AnnoSep = 20;
|
|
const char AnnoDurMark = 21;
|
|
const char AnnoEnd = 0;
|
|
|
|
/*! \struct EDFHeader
|
|
\brief Represents the EDF+ header structure, used as a place holder while processing the text data.
|
|
\note More information on the EDF+ file format can be obtained from http://edfplus.info
|
|
*/
|
|
struct EDFHeaderRaw {
|
|
char version[8];
|
|
char patientident[80];
|
|
char recordingident[80];
|
|
char datetime[16];
|
|
char num_header_bytes[8];
|
|
char reserved[44];
|
|
char num_data_records[8];
|
|
char dur_data_records[8];
|
|
char num_signals[4];
|
|
}
|
|
#ifndef _MSC_VER
|
|
__attribute__((packed))
|
|
#endif
|
|
;
|
|
const int EDFHeaderSize = sizeof(EDFHeaderRaw);
|
|
|
|
/*! \struct EDFHeaderQT
|
|
\brief Contains the QT version of the EDF header information
|
|
*/
|
|
struct EDFHeaderQT {
|
|
public:
|
|
long version;
|
|
QString patientident;
|
|
QString recordingident;
|
|
QDateTime startdate_orig;
|
|
long num_header_bytes;
|
|
QString reserved44;
|
|
long num_data_records;
|
|
double duration_Seconds;
|
|
int num_signals;
|
|
};
|
|
|
|
/*! \struct EDFSignal
|
|
\brief Contains information about a single EDF+ Signal
|
|
\note More information on the EDF+ file format can be obtained from http://edfplus.info
|
|
*/
|
|
struct EDFSignal {
|
|
public:
|
|
QString label; //! \brief Name of this Signal
|
|
QString transducer_type; //! \brief Tranducer Type (source of the data, usually blank)
|
|
QString physical_dimension; //! \brief The units of measurements represented by this signal
|
|
EventDataType physical_minimum; //! \brief The minimum limits of the ungained data
|
|
EventDataType physical_maximum; //! \brief The maximum limits of the ungained data
|
|
EventDataType digital_minimum; //! \brief The minimum limits of the data with gain and offset applied
|
|
EventDataType digital_maximum; //! \brief The maximum limits of the data with gain and offset applied
|
|
EventDataType gain; //! \brief Raw integer data is multiplied by this value
|
|
EventDataType offset; //! \brief This value is added to the raw data
|
|
QString prefiltering; //! \brief Any prefiltering methods used (usually blank)
|
|
long sampleCnt; //! \brief Number of samples per record
|
|
QString reserved; //! \brief Reserved (usually blank)
|
|
|
|
qint16 * dataArray; //! \brief Offset in record to the signals sample data
|
|
};
|
|
|
|
/*! \class Annotation
|
|
\author Phil Olynyk
|
|
\brief Hold the annotation text from an EDF file
|
|
*/
|
|
class Annotation
|
|
{
|
|
public:
|
|
Annotation() { duration = -1.0; };
|
|
Annotation( double off, double dur, QString tx ) {
|
|
offset = off;
|
|
duration = dur;
|
|
text = tx;
|
|
};
|
|
virtual ~Annotation() {};
|
|
|
|
double offset;
|
|
double duration;
|
|
QString text;
|
|
};
|
|
|
|
/*! \class EDFInfo
|
|
\author Phil Olynyk
|
|
\author Mark Watkins <mark@jedimark.net>
|
|
\brief Parse an EDF+ data file into a list of EDFSignal's
|
|
\note More information on the EDF+ file format can be obtained from http://edfplus.info
|
|
*/
|
|
class EDFInfo
|
|
{
|
|
public:
|
|
//! \brief Constructs an EDFParser object, opening the filename if one supplied
|
|
EDFInfo();
|
|
|
|
virtual ~EDFInfo();
|
|
|
|
virtual QByteArray * Open(const QString & name); //! \brief Open the EDF+ file, and read it's header
|
|
|
|
virtual bool Parse(QByteArray * fileData); //! \brief Parse the EDF+ file into the EDFheaderQT. Must call Open(..) first.
|
|
|
|
virtual EDFSignal *lookupLabel(const QString & name, int index=0); //! \brief Return a ptr to the i'th signal with that name
|
|
|
|
virtual long GetNumSignals() { return edfHdr.num_signals; } //! \brief Returns the number of signals contained in this EDF file
|
|
|
|
virtual long GetNumDataRecords() { return edfHdr.num_data_records; } //! \brief Returns the number of data records contained per signal.
|
|
|
|
virtual double GetDuration() { return edfHdr.duration_Seconds; } //! \brief Returns the duration represented by this EDF file
|
|
|
|
virtual QString GetPatient() { return edfHdr.patientident; } //! \brief Returns the patientid field from the EDF header
|
|
|
|
// The data members follow
|
|
|
|
QString filename; //! \brief For debug and error messages
|
|
|
|
EDFHeaderQT edfHdr; //! \brief The header in a QT friendly form
|
|
|
|
QVector<EDFSignal> edfsignals; //! \brief Holds the EDFSignals contained in this edf file
|
|
|
|
// QVector< QVector<qint16> > dataRecords; //! \brief Holds the datarecords
|
|
|
|
QVector< QVector<Annotation> > annotations; //! \brief Holds the Annotaions for this EDF file
|
|
|
|
QStringList signal_labels; //! \brief An by-name indexed into the EDFSignal data
|
|
|
|
QHash<QString, QList<EDFSignal *> > signalList; //! \brief ResMed sometimes re-uses the SAME signal name
|
|
|
|
// the following could be private
|
|
private:
|
|
QVector<Annotation> ReadAnnotations( const char * data, int charLen ); //! \brief Create an Annotaion vector from the signal values
|
|
|
|
QString ReadBytes(unsigned n); //! \brief Read n bytes of 8 bit data from the EDF+ data stream
|
|
|
|
qint16 Read16(); //! \brief Read 16 bit word of data from the EDF+ data stream
|
|
|
|
//! \brief This is the array holding the EDF file data
|
|
QByteArray * fileData;
|
|
//! \brief The EDF+ files header structure, used as a place holder while processing the text data.
|
|
EDFHeaderRaw *hdrPtr;
|
|
//! \brief This is the array of signal descriptors and values
|
|
char *signalPtr;
|
|
|
|
long filesize;
|
|
long datasize;
|
|
long pos;
|
|
bool eof;
|
|
};
|
|
|
|
|
|
#endif // EDFPARSER_H
|