2014-04-09 21:01:57 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
* vim: set ts=8 sts=4 et sw=4 tw=99:
|
|
|
|
*
|
|
|
|
* SleepLib PRS1 Loader Header
|
|
|
|
*
|
|
|
|
* Copyright (c) 2011-2014 Mark Watkins <jedimark@users.sourceforge.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 Linux
|
|
|
|
* distribution for more details. */
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
#ifndef PRS1LOADER_H
|
|
|
|
#define PRS1LOADER_H
|
|
|
|
//#include <map>
|
|
|
|
//using namespace std;
|
|
|
|
#include "SleepLib/machine.h" // Base class: MachineLoader
|
|
|
|
#include "SleepLib/machine_loader.h"
|
|
|
|
#include "SleepLib/profiles.h"
|
|
|
|
|
|
|
|
|
|
|
|
//********************************************************************************************
|
|
|
|
/// IMPORTANT!!!
|
|
|
|
//********************************************************************************************
|
|
|
|
// Please INCREMENT the following value when making changes to this loaders implementation.
|
|
|
|
//
|
2014-07-03 01:59:50 +00:00
|
|
|
const int prs1_data_version = 13;
|
2011-06-26 08:30:44 +00:00
|
|
|
//
|
|
|
|
//********************************************************************************************
|
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
/*! \class PRS1
|
|
|
|
\brief PRS1 customized machine object (via CPAP)
|
|
|
|
*/
|
2014-04-17 05:58:57 +00:00
|
|
|
class PRS1: public CPAP
|
2011-06-26 08:30:44 +00:00
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
public:
|
2014-07-11 12:09:38 +00:00
|
|
|
PRS1(MachineID id = 0);
|
2011-06-26 08:30:44 +00:00
|
|
|
virtual ~PRS1();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
const int max_load_buffer_size = 1024 * 1024;
|
2014-05-31 21:25:07 +00:00
|
|
|
const QString prs1_class_name = STR_MACH_PRS1;
|
|
|
|
|
|
|
|
/*! \struct PRS1Waveform
|
|
|
|
\brief Used in PRS1 Waveform Parsing */
|
|
|
|
struct PRS1Waveform {
|
|
|
|
PRS1Waveform(quint16 i, quint8 f) {
|
|
|
|
interleave = i;
|
|
|
|
sample_format = f;
|
|
|
|
}
|
|
|
|
quint16 interleave;
|
|
|
|
quint8 sample_format;
|
|
|
|
};
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
|
2014-05-31 21:25:07 +00:00
|
|
|
class PRS1DataChunk
|
|
|
|
{
|
|
|
|
friend class PRS1DataGroup;
|
|
|
|
public:
|
|
|
|
PRS1DataChunk() {
|
|
|
|
timestamp = 0;
|
|
|
|
ext = 255;
|
|
|
|
sessionid = 0;
|
|
|
|
htype = 0;
|
|
|
|
family = 0;
|
|
|
|
familyVersion = 0;
|
|
|
|
duration = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
~PRS1DataChunk() {
|
|
|
|
}
|
|
|
|
inline int size() const { return m_data.size(); }
|
|
|
|
|
|
|
|
QByteArray m_data;
|
|
|
|
|
|
|
|
SessionID sessionid;
|
|
|
|
|
|
|
|
quint8 fileVersion;
|
|
|
|
quint8 ext;
|
|
|
|
quint8 htype;
|
|
|
|
quint8 family;
|
|
|
|
quint8 familyVersion;
|
|
|
|
quint32 timestamp;
|
|
|
|
|
|
|
|
quint16 duration;
|
|
|
|
|
|
|
|
QList<PRS1Waveform> waveformInfo;
|
|
|
|
};
|
|
|
|
|
|
|
|
class PRS1SessionData
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
PRS1SessionData() {
|
|
|
|
compliance = summary = event = nullptr;
|
|
|
|
session = nullptr;
|
|
|
|
}
|
|
|
|
PRS1SessionData(const PRS1SessionData & copy) {
|
|
|
|
session = copy.session;
|
|
|
|
compliance = copy.compliance;
|
|
|
|
summary = copy.summary;
|
|
|
|
event = copy.event;
|
|
|
|
waveforms = copy.waveforms;
|
|
|
|
}
|
|
|
|
~PRS1SessionData() {
|
|
|
|
delete compliance;
|
|
|
|
delete summary;
|
|
|
|
delete event;
|
|
|
|
Q_FOREACH(PRS1DataChunk * c, waveforms) {
|
|
|
|
delete c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ParseCompliance();
|
|
|
|
bool ParseSummary();
|
|
|
|
bool ParseEvents();
|
|
|
|
bool ParseWaveforms();
|
|
|
|
|
|
|
|
//! \brief Parse a single data chunk from a .002 file containing event data for a standard system one machine
|
|
|
|
bool ParseF0Events();
|
|
|
|
|
|
|
|
//! \brief Parse a single data chunk from a .002 file containing event data for a family 5 ASV machine (which has a different format)
|
|
|
|
bool ParseF5Events();
|
|
|
|
|
|
|
|
Session * session;
|
|
|
|
|
|
|
|
PRS1DataChunk * compliance;
|
|
|
|
PRS1DataChunk * summary;
|
|
|
|
PRS1DataChunk * event;
|
|
|
|
QList<PRS1DataChunk *> waveforms;
|
|
|
|
};
|
|
|
|
|
2014-07-11 06:13:44 +00:00
|
|
|
class PRS1Loader;
|
2014-05-31 21:25:07 +00:00
|
|
|
|
|
|
|
struct PRS1FileGroup
|
|
|
|
{
|
2014-07-11 06:13:44 +00:00
|
|
|
PRS1FileGroup() { loader = NULL; }
|
2014-05-31 21:25:07 +00:00
|
|
|
PRS1FileGroup(const PRS1FileGroup & copy) {
|
|
|
|
compliance = copy.compliance;
|
|
|
|
summary = copy.summary;
|
|
|
|
event = copy.event;
|
|
|
|
waveform = copy.waveform;
|
2014-07-11 06:13:44 +00:00
|
|
|
loader = copy.loader;
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
|
|
|
~PRS1FileGroup() {
|
|
|
|
}
|
|
|
|
|
|
|
|
QString compliance;
|
|
|
|
QString summary;
|
|
|
|
QString event;
|
|
|
|
QString waveform;
|
|
|
|
|
|
|
|
bool ParseFile(QString path);
|
2014-07-11 06:13:44 +00:00
|
|
|
void ParseChunks(PRS1Loader *);
|
|
|
|
|
|
|
|
PRS1Loader * loader;
|
2014-05-31 21:25:07 +00:00
|
|
|
|
|
|
|
QMap<SessionID, PRS1SessionData*> sessions;
|
|
|
|
};
|
|
|
|
|
|
|
|
class PRS1Import:public ImportTask
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
PRS1Import(PRS1Loader * l, SessionID s, PRS1FileGroup *g, Machine * m): loader(l), sessionid(s), group(g), mach(m) {}
|
|
|
|
virtual ~PRS1Import() {}
|
|
|
|
virtual void run();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
PRS1Loader * loader;
|
|
|
|
SessionID sessionid;
|
|
|
|
PRS1FileGroup *group;
|
|
|
|
Machine * mach;
|
|
|
|
};
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
/*! \class PRS1Loader
|
|
|
|
\brief Philips Respironics System One Loader Module
|
|
|
|
*/
|
2011-06-26 08:30:44 +00:00
|
|
|
class PRS1Loader : public MachineLoader
|
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
public:
|
2011-06-26 08:30:44 +00:00
|
|
|
PRS1Loader();
|
|
|
|
virtual ~PRS1Loader();
|
2014-04-26 09:54:08 +00:00
|
|
|
|
|
|
|
//! \brief Detect if the given path contains a valid Folder structure
|
|
|
|
virtual bool Detect(const QString & path);
|
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Scans directory path for valid PRS1 signature
|
2014-07-11 12:09:38 +00:00
|
|
|
virtual int Open(QString path);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Returns the database version of this loader
|
2011-07-15 13:30:41 +00:00
|
|
|
virtual int Version() { return prs1_data_version; }
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
//! \brief Return the loaderName, in this case "PRS1"
|
|
|
|
virtual const QString &loaderName() { return prs1_class_name; }
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
// //! \brief Create a new PRS1 machine record, indexed by Serial number.
|
|
|
|
//Machine *CreateMachine(QString serial);
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Register this Module to the list of Loaders, so it knows to search for PRS1 data.
|
2011-06-26 08:30:44 +00:00
|
|
|
static void Register();
|
2014-07-11 06:13:44 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
virtual MachineInfo newInfo() {
|
|
|
|
return MachineInfo(MT_CPAP, prs1_class_name, QObject::tr("Philips Respironics"), QString(), QString(), QString(), QObject::tr("System One"), QDateTime::currentDateTime(), prs1_data_version);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-11 06:13:44 +00:00
|
|
|
QHash<SessionID, PRS1FileGroup*> prs1sessions;
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
protected:
|
2011-06-26 08:30:44 +00:00
|
|
|
QString last;
|
2014-04-17 05:58:57 +00:00
|
|
|
QHash<QString, Machine *> PRS1List;
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Opens the SD folder structure for this machine, scans for data files and imports any new sessions
|
2014-07-11 12:09:38 +00:00
|
|
|
int OpenMachine(Machine *m, QString path);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Parses "properties.txt" file containing machine information
|
2014-04-17 05:58:57 +00:00
|
|
|
bool ParseProperties(Machine *m, QString filename);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2011-12-10 12:14:48 +00:00
|
|
|
//bool OpenSummary(Session *session,QString filename);
|
|
|
|
//bool OpenEvents(Session *session,QString filename);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Parse a .005 waveform file, extracting Flow Rate waveform (and Mask Pressure data if available)
|
2011-12-10 12:14:48 +00:00
|
|
|
bool OpenWaveforms(SessionID sid, QString filename);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2012-01-09 15:38:41 +00:00
|
|
|
// //! \brief ParseWaveform chunk.. Currently unused, as the old one works fine.
|
|
|
|
//bool ParseWaveform(qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, quint16 duration, quint16 num_signals, quint16 interleave, quint8 sample_format);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Parse a data chunk from the .000 (brick) and .001 (summary) files.
|
2014-04-17 05:58:57 +00:00
|
|
|
bool ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, unsigned char *data,
|
|
|
|
quint16 size, int family, int familyVersion);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
|
2011-12-10 12:14:48 +00:00
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Open a PRS1 data file, and break into data chunks, delivering them to the correct parser.
|
2011-12-10 12:14:48 +00:00
|
|
|
bool OpenFile(Machine *mach, QString filename);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2011-12-10 12:14:48 +00:00
|
|
|
//bool Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp,long fpos);
|
|
|
|
//bool Parse002ASV(Session *session,unsigned char *buffer,int size,qint64 timestamp,long fpos);
|
2014-04-17 05:58:57 +00:00
|
|
|
unsigned char *m_buffer;
|
2011-12-06 14:39:14 +00:00
|
|
|
QHash<SessionID, Session *> extra_session;
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief PRS1 Data files can store multiple sessions, so store them in this list for later processing.
|
2011-12-10 12:14:48 +00:00
|
|
|
QHash<SessionID, Session *> new_sessions;
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2011-12-10 12:14:48 +00:00
|
|
|
qint32 summary_duration;
|
|
|
|
};
|
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
#endif // PRS1LOADER_H
|