mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 02:30:44 +00:00
Add initial version of Lövenstein Prisma loader
This commit is contained in:
parent
0001681146
commit
f2b811e442
@ -163,6 +163,7 @@ const QString STR_MACH_SleepStyle = "SleepStyle";
|
||||
const QString STR_MACH_MSeries = "MSeries";
|
||||
const QString STR_MACH_CMS50 = "CMS50";
|
||||
const QString STR_MACH_ZEO = "Zeo";
|
||||
const QString STR_MACH_Prisma = "Prisma";
|
||||
|
||||
const QString STR_PREF_Language = "Language";
|
||||
|
||||
|
699
oscar/SleepLib/loader_plugins/prisma_loader.cpp
Normal file
699
oscar/SleepLib/loader_plugins/prisma_loader.cpp
Normal file
@ -0,0 +1,699 @@
|
||||
/* SleepLib Prisma Loader Implementation
|
||||
*
|
||||
* Copyright (c) 2019-2022 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. */
|
||||
|
||||
#include <QApplication>
|
||||
#include <QString>
|
||||
#include <QDateTime>
|
||||
#include <QDir>
|
||||
#include <QFile>
|
||||
#include <QBuffer>
|
||||
#include <QDataStream>
|
||||
#include <QMessageBox>
|
||||
#include <QJsonDocument>
|
||||
#include <QJsonObject>
|
||||
#include <QDebug>
|
||||
|
||||
#include <cmath>
|
||||
|
||||
#include "SleepLib/schema.h"
|
||||
#include "SleepLib/importcontext.h"
|
||||
#include "prisma_loader.h"
|
||||
#include "SleepLib/session.h"
|
||||
#include "SleepLib/calcs.h"
|
||||
#include "rawdata.h"
|
||||
|
||||
#define CONFIG_FILE "config.pscfg"
|
||||
|
||||
//********************************************************************************************
|
||||
/// IMPORTANT!!!
|
||||
//********************************************************************************************
|
||||
// Please INCREMENT the prisma_data_version in prisma_loader.h when making changes to this
|
||||
// loader that change loader behaviour or modify channels.
|
||||
//********************************************************************************************
|
||||
|
||||
// parameters
|
||||
ChannelID Prisma_Mode = 0, Prisma_SoftPAP = 0, Prisma_PSoft = 0, Prisma_PSoft_Min = 0, Prisma_AutoStart = 0, Prisma_Softstart_Time = 0, Prisma_Softstart_TimeMax = 0, Prisma_TubeType = 0, Prisma_PMaxOA = 0;
|
||||
// waveforms
|
||||
ChannelID Prisma_ObstructLevel = 0, Prisma_rMVFluctuation = 0, Prisma_rRMV= 0, Prisma_PressureMeasured = 0, Prisma_FlowFull = 0, Prisma_SPRStatus = 0;
|
||||
// events
|
||||
ChannelID Prisma_Artifact = 0, Prisma_CriticalLeak = 0, Prisma_eSO = 0, Prisma_eMO = 0, Prisma_eS = 0, Prisma_eF = 0, Prisma_DeepSleep = 0;
|
||||
|
||||
QString PrismaLoader::PresReliefLabel() { return QString("SoftPAP: "); }
|
||||
ChannelID PrismaLoader::PresReliefMode() { return Prisma_SoftPAP; }
|
||||
ChannelID PrismaLoader::CPAPModeChannel() { return Prisma_Mode; }
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
bool WMEDFInfo::ParseSignalData() {
|
||||
// Now check the file isn't truncated before allocating space for the values
|
||||
long allocsize = 0;
|
||||
for (auto & sig : edfsignals) {
|
||||
if (edfHdr.num_data_records > 0) {
|
||||
allocsize += sig.sampleCnt * edfHdr.num_data_records * 2;
|
||||
}
|
||||
}
|
||||
// allocate the arrays for the signal values
|
||||
for (auto & sig : edfsignals) {
|
||||
long samples = sig.sampleCnt * edfHdr.num_data_records;
|
||||
if (edfHdr.num_data_records <= 0) {
|
||||
sig.dataArray = nullptr;
|
||||
continue;
|
||||
}
|
||||
sig.dataArray = new qint16 [samples];
|
||||
}
|
||||
for (int recNo = 0; recNo < edfHdr.num_data_records; recNo++) {
|
||||
for (auto & sig : edfsignals) {
|
||||
for (int j=0;j<sig.sampleCnt;j++) {
|
||||
if (sig.reserved == "#1") {
|
||||
if (sig.digital_minimum >= 0)
|
||||
{
|
||||
sig.dataArray[recNo*sig.sampleCnt+j]=(qint16)Read8U();
|
||||
}
|
||||
else
|
||||
{
|
||||
sig.dataArray[recNo*sig.sampleCnt+j]=(qint16)Read8S();
|
||||
}
|
||||
} else if (sig.reserved == "#2") {
|
||||
qint16 t=Read16();
|
||||
sig.dataArray[recNo*sig.sampleCnt+j]=t;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
qint8 WMEDFInfo::Read8S()
|
||||
{
|
||||
if ((pos + 1) > datasize) {
|
||||
eof = true;
|
||||
return 0;
|
||||
}
|
||||
qint8 res = *(qint8 *)&signalPtr[pos];
|
||||
pos += 1;
|
||||
return res;
|
||||
}
|
||||
|
||||
quint8 WMEDFInfo::Read8U()
|
||||
{
|
||||
if ((pos + 1) > datasize) {
|
||||
eof = true;
|
||||
return 0;
|
||||
}
|
||||
quint8 res = *(quint8 *)&signalPtr[pos];
|
||||
pos += 1;
|
||||
return res;
|
||||
}
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
void PrismaImport::run()
|
||||
{
|
||||
qDebug() << "PRISMA IMPORT" << eventFileName << " " << signalFileName;
|
||||
|
||||
if (!wmedf.Open(signalFileName)) {
|
||||
qWarning() << "Signal file open failed" << signalFileName;
|
||||
return;
|
||||
}
|
||||
|
||||
if (!wmedf.Parse()) {
|
||||
qWarning() << "Signal file parsing failed" << signalFileName;
|
||||
return;
|
||||
}
|
||||
|
||||
eventFile = new PrismaEventFile(eventFileName);
|
||||
|
||||
startdate = qint64(wmedf.edfHdr.startdate_orig.toTime_t()) * 1000L;
|
||||
enddate = startdate + wmedf.GetDuration() * qint64(wmedf.GetNumDataRecords()) * 1000;
|
||||
|
||||
session = loader->context()->CreateSession(sessionid);
|
||||
session->really_set_first(startdate);
|
||||
session->really_set_last(enddate);
|
||||
|
||||
// TODO AXT: set physical limits from a config file
|
||||
session->setPhysMax(CPAP_Pressure, 20);
|
||||
session->setPhysMin(CPAP_Pressure, 4);
|
||||
session->setPhysMax(CPAP_IPAP, 20);
|
||||
session->setPhysMin(CPAP_IPAP, 4);
|
||||
session->setPhysMax(CPAP_EPAP, 20);
|
||||
session->setPhysMin(CPAP_EPAP, 4);
|
||||
|
||||
// set session parameters
|
||||
auto parameters = eventFile->getParameters();
|
||||
// TODO AXT: extract
|
||||
switch(parameters[PRISMA_MODE]) {
|
||||
case PRISMA_MODE_CPAP:
|
||||
session->settings[CPAP_Mode] = (int)MODE_CPAP;
|
||||
session->settings[Prisma_Mode] = (int)PRISMA_COMBINED_MODE_CPAP;
|
||||
break;
|
||||
case PRISMA_MODE_APAP:
|
||||
session->settings[CPAP_Mode] = (int)MODE_APAP;
|
||||
switch (parameters[PRISMA_APAP_DYNAMIC])
|
||||
{
|
||||
case PRISMA_APAP_MODE_STANDARD:
|
||||
session->settings[Prisma_Mode] = (int)PRISMA_COMBINED_MODE_APAP_STD;
|
||||
break;
|
||||
case PRISMA_APAP_MODE_DYNAMIC:
|
||||
session->settings[Prisma_Mode] = (int)PRISMA_COMBINED_MODE_APAP_DYN;
|
||||
break;
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
session->settings[CPAP_PressureMin] = parameters[PRISMA_PRESSURE] / 100;
|
||||
session->settings[CPAP_PressureMax] = parameters[PRISMA_PRESSURE_MAX] / 100;
|
||||
session->settings[Prisma_SoftPAP] = parameters[PRISMA_SOFTPAP];
|
||||
session->settings[Prisma_PSoft] = parameters[PRISMA_PSOFT] / 100.0;
|
||||
session->settings[Prisma_PSoft_Min] = parameters[PRISMA_PSOFT_MIN] / 100;
|
||||
session->settings[Prisma_AutoStart] = parameters[PRISMA_AUTOSTART] / 100;
|
||||
session->settings[Prisma_Softstart_Time] = parameters[PRISMA_SOFTSTART_TIME];
|
||||
session->settings[Prisma_Softstart_TimeMax] = parameters[PRISMA_SOFTSTART_TIME_MAX];
|
||||
session->settings[Prisma_TubeType] = parameters[PRISMA_TUBE_TYPE];
|
||||
session->settings[Prisma_PMaxOA] = parameters[PRISMA_PMAXOA] / 100;
|
||||
|
||||
// add waveforms
|
||||
AddWaveform(CPAP_Pressure, QString("CPAPPressure"));
|
||||
AddWaveform(CPAP_EPAP, QString("EPAP"));
|
||||
AddWaveform(CPAP_MaskPressure, QString("Pressure"));
|
||||
AddWaveform(CPAP_FlowRate, QString("RespFlow"));
|
||||
AddWaveform(CPAP_Leak, QString("LeakFlowBreath"));
|
||||
AddWaveform(Prisma_ObstructLevel, QString("ObstructLevel"));
|
||||
AddWaveform(Prisma_rMVFluctuation, QString("rMVFluctuation"));
|
||||
AddWaveform(Prisma_rRMV, QString("rRMV"));
|
||||
AddWaveform(Prisma_PressureMeasured, QString("PressureMeasured"));
|
||||
AddWaveform(Prisma_FlowFull, QString("FlowFull"));
|
||||
AddWaveform(Prisma_SPRStatus, QString("SPRStatus"));
|
||||
|
||||
// add signals
|
||||
AddEvents(CPAP_Obstructive, PRISMA_EVENT_OBSTRUCTIVE_APNEA);
|
||||
AddEvents(CPAP_ClearAirway, PRISMA_EVENT_CENTRAL_APNEA);
|
||||
AddEvents(CPAP_Apnea, { PRISMA_EVENT_APNEA_LEAKAGE, PRISMA_EVENT_APNEA_HIGH_PRESSURE, PRISMA_EVENT_APNEA_MOVEMENT});
|
||||
AddEvents(CPAP_Hypopnea, { PRISMA_EVENT_OBSTRUCTIVE_HYPOPNEA, PRISMA_EVENT_CENTRAL_HYPOPNEA});
|
||||
AddEvents(CPAP_RERA, PRISMA_EVENT_RERA);
|
||||
AddEvents(CPAP_Snore, PRISMA_EVENT_SNORE);
|
||||
AddEvents(CPAP_CSR, PRISMA_EVENT_CS_RESPIRATION);
|
||||
AddEvents(CPAP_FlowLimit, PRISMA_EVENT_FLOW_LIMITATION);
|
||||
|
||||
AddEvents(Prisma_Artifact, PRISMA_EVENT_ARTIFACT);
|
||||
AddEvents(Prisma_CriticalLeak, PRISMA_EVENT_CRITICAL_LEAKAGE);
|
||||
AddEvents(Prisma_eSO, PRISMA_EVENT_EPOCH_SEVERE_OBSTRUCTION);
|
||||
AddEvents(Prisma_eMO, PRISMA_EVENT_EPOCH_MILD_OBSTRUCTION);
|
||||
AddEvents(Prisma_eF, PRISMA_EVENT_EPOCH_FLOW_LIMITATION);
|
||||
AddEvents(Prisma_eS, PRISMA_EVENT_EPOCH_SNORE);
|
||||
AddEvents(Prisma_DeepSleep, PRISMA_EVENT_EPOCH_DEEPSLEEP);
|
||||
|
||||
session->SetChanged(true);
|
||||
loader->context()->AddSession(session);
|
||||
}
|
||||
|
||||
void PrismaImport::AddWaveform(ChannelID code, QString edfLabel)
|
||||
{
|
||||
EDFSignal * es = wmedf.lookupLabel(edfLabel);
|
||||
if (es != nullptr) {
|
||||
qint64 duration = wmedf.GetNumDataRecords() * wmedf.GetDuration() * 1000L;
|
||||
long recs = es->sampleCnt * wmedf.GetNumDataRecords();
|
||||
|
||||
double rate = double(duration) / double(recs);
|
||||
EventList *a = session->AddEventList(code, EVL_Waveform, es->gain, es->offset, 0, 0, rate);
|
||||
a->setDimension(es->physical_dimension);
|
||||
a->AddWaveform(startdate, es->dataArray, recs, duration);
|
||||
|
||||
session->setPhysMin(code, es->physical_minimum);
|
||||
session->setPhysMax(code, es->physical_maximum);
|
||||
}
|
||||
}
|
||||
|
||||
void PrismaImport::AddEvents(ChannelID channel, QList<Prisma_Event_Type> eventTypes)
|
||||
{
|
||||
EventList *eventList = nullptr;
|
||||
for (auto eventType : eventTypes) {
|
||||
QList<PrismaEvent> events = eventFile->getEvents(eventType);
|
||||
for (auto event: events) {
|
||||
if (eventList == nullptr) {
|
||||
eventList = session->AddEventList(channel, EVL_Event, 1.0, 0.0, 0.0, 0.0, 0.0, true);
|
||||
}
|
||||
eventList ->AddEvent(startdate + event.endTime(), event.duration(), event.strength());
|
||||
}
|
||||
}
|
||||
session->AddEventList(channel, EVL_Event);
|
||||
}
|
||||
|
||||
//********************************************************************************************
|
||||
PrismaEventFile::PrismaEventFile(QString fname)
|
||||
{
|
||||
QFile file(fname);
|
||||
QDomDocument dom;
|
||||
|
||||
if(file.open(QIODevice::ReadOnly)) {
|
||||
dom.setContent(&file);
|
||||
file.close();
|
||||
|
||||
QDomElement root = dom.documentElement();
|
||||
|
||||
QDomNodeList deviceEventNodelist = root.elementsByTagName("DeviceEvent");
|
||||
for(int i=0; i < deviceEventNodelist.count(); i++)
|
||||
{
|
||||
QDomElement node=deviceEventNodelist.item(i).toElement();
|
||||
int eventId = node.attribute("DeviceEventID").toInt();
|
||||
if (eventId == 0) {
|
||||
int parameterId = node.attribute("ParameterID").toInt();
|
||||
int value = node.attribute("NewValue").toInt();
|
||||
m_parameters[parameterId] = value;
|
||||
}
|
||||
}
|
||||
|
||||
QDomNodeList respEventNodelist = root.elementsByTagName("RespEvent");
|
||||
for(int i=0; i < respEventNodelist.count(); i++)
|
||||
{
|
||||
QDomElement node=respEventNodelist.item(i).toElement();
|
||||
int eventId = node.attribute("RespEventID").toInt();
|
||||
const int time_quantum = 10;
|
||||
int endTime = node.attribute("EndTime").toInt() * 1000 / time_quantum;
|
||||
int duration = node.attribute("Duration").toInt() / time_quantum;
|
||||
int pressure = node.attribute("Pressure").toInt();
|
||||
int strength = node.attribute("Strength").toInt();
|
||||
m_events[eventId].append(PrismaEvent(endTime, duration, pressure, strength));
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
struct PrismaTestedModel
|
||||
{
|
||||
QString deviceId;
|
||||
const char* name;
|
||||
};
|
||||
|
||||
static const PrismaTestedModel s_PrismaTestedModels[] = {
|
||||
{ "0x92", "Prisma Smart" },
|
||||
{ "", ""}
|
||||
};
|
||||
|
||||
PrismaModelInfo s_PrismaModelInfo;
|
||||
|
||||
PrismaModelInfo::PrismaModelInfo ()
|
||||
{
|
||||
for (int i = 0; !s_PrismaTestedModels[i].deviceId.isEmpty(); i++) {
|
||||
const PrismaTestedModel & model = s_PrismaTestedModels[i];
|
||||
m_modelNames[model.deviceId] = model.name;
|
||||
}
|
||||
}
|
||||
|
||||
bool PrismaModelInfo::IsTested(const QString & deviceId) const
|
||||
{
|
||||
return m_modelNames.contains(deviceId);
|
||||
};
|
||||
|
||||
const char* PrismaModelInfo::Name(const QString & deviceId) const
|
||||
{
|
||||
const char* name;
|
||||
if (m_modelNames.contains(deviceId)) {
|
||||
name = m_modelNames[deviceId];
|
||||
} else {
|
||||
name = "Unknown Model";
|
||||
}
|
||||
return name;
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
PrismaLoader::PrismaLoader()
|
||||
{
|
||||
m_type = MT_CPAP;
|
||||
}
|
||||
|
||||
PrismaLoader::~PrismaLoader()
|
||||
{
|
||||
}
|
||||
|
||||
bool PrismaLoader::Detect(const QString & selectedPath)
|
||||
{
|
||||
QFile configFile(selectedPath + QDir::separator() + CONFIG_FILE);
|
||||
return configFile.exists();
|
||||
}
|
||||
|
||||
int PrismaLoader::Open(const QString & selectedPath)
|
||||
{
|
||||
if (m_ctx == nullptr) {
|
||||
qWarning() << "PrismaLoader::Open() called without a valid m_ctx object present";
|
||||
return 0;
|
||||
}
|
||||
Q_ASSERT(m_ctx);
|
||||
|
||||
qDebug() << "Prisma opening" << selectedPath;
|
||||
|
||||
QString configFilePath = selectedPath + QDir::separator() + CONFIG_FILE;
|
||||
QFile configFile(configFilePath);
|
||||
if (!configFile.exists()) // TODO AXT || !configFile.isReadable() fails
|
||||
{
|
||||
qDebug() << "Prisma config file error" << configFile << " " << configFile.exists() << " " << configFile.isReadable();
|
||||
return 0;
|
||||
}
|
||||
|
||||
m_abort = false;
|
||||
emit setProgressValue(0);
|
||||
emit updateMessage(QObject::tr("Getting Ready..."));
|
||||
QCoreApplication::processEvents();
|
||||
|
||||
MachineInfo info = PeekInfoFromConfig(configFilePath);
|
||||
qDebug() << "Prisma machine info" << info.serial;
|
||||
|
||||
if (info.type == MT_UNKNOWN) {
|
||||
emit deviceIsUnsupported(info);
|
||||
return -1;
|
||||
}
|
||||
|
||||
m_ctx->CreateMachineFromInfo(info);
|
||||
|
||||
if (!s_PrismaModelInfo.IsTested(info.modelnumber)) {
|
||||
qDebug() << info.modelnumber << "untested";
|
||||
emit deviceIsUntested(info);
|
||||
}
|
||||
|
||||
emit updateMessage(QObject::tr("Backing Up Files..."));
|
||||
QCoreApplication::processEvents();
|
||||
|
||||
|
||||
QString backupPath = context()->GetBackupPath() + selectedPath.section("/", -1);
|
||||
if (QDir::cleanPath(selectedPath).compare(QDir::cleanPath(backupPath)) != 0) {
|
||||
copyPath(selectedPath, backupPath);
|
||||
}
|
||||
|
||||
emit updateMessage(QObject::tr("Scanning Files..."));
|
||||
QCoreApplication::processEvents();
|
||||
|
||||
// TODO AXT extract
|
||||
char out[12];
|
||||
int serialInDecimal;
|
||||
sscanf(info.serial.toLocal8Bit().data() , "%x", &serialInDecimal);
|
||||
snprintf(out, 12, "%010d", serialInDecimal);
|
||||
|
||||
ScanFiles(info, selectedPath + QDir::separator() + out);
|
||||
|
||||
int tasks = countTasks();
|
||||
|
||||
emit updateMessage(QObject::tr("Importing Sessions..."));
|
||||
QCoreApplication::processEvents();
|
||||
|
||||
runTasks(AppSetting->multithreading());
|
||||
|
||||
m_ctx->FlushUnexpectedMessages();
|
||||
|
||||
return tasks;
|
||||
}
|
||||
|
||||
MachineInfo PrismaLoader::PeekInfo(const QString & selectedPath)
|
||||
{
|
||||
qDebug() << "PeekInfo " << selectedPath;
|
||||
if (!Detect(selectedPath))
|
||||
return MachineInfo();
|
||||
|
||||
return PeekInfoFromConfig(selectedPath + QDir::separator() + CONFIG_FILE);
|
||||
}
|
||||
|
||||
MachineInfo PrismaLoader::PeekInfoFromConfig(const QString & configFilePath)
|
||||
{
|
||||
QFile configFile(configFilePath);
|
||||
|
||||
if (configFile.exists()) {
|
||||
if (!configFile.open(QIODevice::ReadOnly)) {
|
||||
return MachineInfo();
|
||||
}
|
||||
MachineInfo info = newInfo();
|
||||
QByteArray configData = configFile.readAll();
|
||||
configFile.close();
|
||||
|
||||
QJsonDocument configDoc(QJsonDocument::fromJson(configData));
|
||||
QJsonObject configObj = configDoc.object();
|
||||
QJsonObject devObj = configObj["dev"].toObject();
|
||||
info.modelnumber=configObj["devid"].toString();
|
||||
info.serial = devObj["sn"].toString();
|
||||
// TODO AXT load props
|
||||
info.properties["cica"] = "mica";
|
||||
return info;
|
||||
}
|
||||
return MachineInfo();
|
||||
}
|
||||
|
||||
void PrismaLoader::ScanFiles(const MachineInfo& info, const QString & machinePath)
|
||||
{
|
||||
Q_ASSERT(m_ctx);
|
||||
qDebug() << "SCANFILES" << machinePath;
|
||||
|
||||
QDir machineDir(machinePath);
|
||||
machineDir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs | QDir::NoSymLinks);
|
||||
machineDir.setSorting(QDir::Name);
|
||||
QFileInfoList dayListing = machineDir.entryInfoList();
|
||||
|
||||
QSet<SessionID> sessions;
|
||||
QHash<SessionID, QString> eventFiles;
|
||||
QHash<SessionID, QString> signalFiles;
|
||||
|
||||
qint64 ignoreBefore = m_ctx->IgnoreSessionsOlderThan().toMSecsSinceEpoch()/1000;
|
||||
bool ignoreOldSessions = m_ctx->ShouldIgnoreOldSessions();
|
||||
|
||||
qDebug() << "INFO " << ignoreBefore << " " << ignoreOldSessions;
|
||||
|
||||
for (auto & dayDirInfo : dayListing) {
|
||||
QDir dayDir(dayDirInfo.canonicalFilePath());
|
||||
dayDir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs | QDir::NoSymLinks);
|
||||
dayDir.setSorting(QDir::Name);
|
||||
QFileInfoList subDirs = dayDir.entryInfoList();
|
||||
QDir dataDir;
|
||||
|
||||
if (subDirs.size() == 0) {
|
||||
dataDir = dayDir;
|
||||
} else if (subDirs.size() == 1) {
|
||||
dataDir = QDir(subDirs.at(0).canonicalFilePath());
|
||||
} else {
|
||||
qWarning() << "PrismaLoader: Directory structure not recognized!";
|
||||
continue;
|
||||
}
|
||||
|
||||
dataDir.setFilter(QDir::NoDotAndDotDot | QDir::Files | QDir::NoSymLinks);
|
||||
dataDir.setSorting(QDir::Name);
|
||||
|
||||
if (dataDir.exists()) {
|
||||
for (auto & inputFile : dataDir.entryInfoList()) {
|
||||
QString fileName = inputFile.fileName().toLower();
|
||||
if (fileName.startsWith("event_") && fileName.endsWith(".xml")) {
|
||||
SessionID sid = fileName.mid(6,fileName.size()-4-6).toLong();
|
||||
sessions += sid;
|
||||
eventFiles[sid] = inputFile.canonicalFilePath();
|
||||
}
|
||||
if (inputFile.fileName().toLower().startsWith("signal_") && inputFile.fileName().toLower().endsWith(".wmedf")) {
|
||||
SessionID sid = fileName.mid(7,fileName.size()-6-7).toLong();
|
||||
sessions += sid;
|
||||
signalFiles[sid] = inputFile.canonicalFilePath();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for(auto & sid : sessions) {
|
||||
queTask(new PrismaImport(this, info, sid, eventFiles[sid], signalFiles[sid]));
|
||||
}
|
||||
}
|
||||
|
||||
using namespace schema;
|
||||
|
||||
void PrismaLoader::initChannels()
|
||||
{
|
||||
Channel * chan = nullptr;
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_Mode=0xe400, SETTING, MT_CPAP, SESSION,
|
||||
"PrismaMode", QObject::tr("Mode"),
|
||||
QObject::tr("PAP Mode"),
|
||||
QObject::tr("PAP Mode"),
|
||||
"", LOOKUP, Qt::green));
|
||||
chan->addOption(PRISMA_COMBINED_MODE_CPAP, QObject::tr("CPAP"));
|
||||
chan->addOption(PRISMA_COMBINED_MODE_APAP_STD, QObject::tr("APAP (std)"));
|
||||
chan->addOption(PRISMA_COMBINED_MODE_APAP_DYN, QObject::tr("APAP (dyn)"));
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_SoftPAP=0xe401, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_SoftPAP",
|
||||
QObject::tr("SoftPAP Mode"),
|
||||
QObject::tr("SoftPAP Mode"),
|
||||
QObject::tr("SoftPAP Mode"),
|
||||
"", LOOKUP, Qt::green));
|
||||
chan->addOption(Prisma_SoftPAP_OFF, QObject::tr("Off"));
|
||||
chan->addOption(Prisma_SoftPAP_SLIGHT, QObject::tr("Slight"));
|
||||
chan->addOption(Prisma_SoftPAP_STANDARD, QObject::tr("Standard"));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_PSoft=0xe402, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_PSoft", QObject::tr("PSoft"),
|
||||
QObject::tr("PSoft"),
|
||||
QObject::tr("PSoft"),
|
||||
STR_UNIT_CMH2O, LOOKUP, Qt::green));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_PSoft_Min=0xe403, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_PSoft_Min", QObject::tr("PSoftMin"),
|
||||
QObject::tr("PSoftMin"),
|
||||
QObject::tr("PSoftMin"),
|
||||
STR_UNIT_CMH2O, LOOKUP, Qt::green));
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_AutoStart=0xe404, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_AutoStart", QObject::tr("AutoStart"),
|
||||
QObject::tr("AutoStart"),
|
||||
QObject::tr("AutoStart"),
|
||||
"", LOOKUP, Qt::green));
|
||||
chan->addOption(0, STR_TR_Off);
|
||||
chan->addOption(1, STR_TR_On);
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_Softstart_Time=0xe405, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_Softstart_Time", QObject::tr("Softstart_Time"),
|
||||
QObject::tr("Softstart_Time"),
|
||||
QObject::tr("Softstart_Time"),
|
||||
STR_UNIT_Minutes, LOOKUP, Qt::green));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_Softstart_TimeMax=0xe406, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_Softstart_TimeMax", QObject::tr("Softstart_TimeMax"),
|
||||
QObject::tr("Softstart_TimeMax"),
|
||||
QObject::tr("Softstart_TimeMax"),
|
||||
STR_UNIT_Minutes, LOOKUP, Qt::green));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_TubeType=0xe407, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_TubeType", QObject::tr("TubeType"),
|
||||
QObject::tr("TubeType"),
|
||||
QObject::tr("TubeType"),
|
||||
STR_UNIT_CM, LOOKUP, Qt::green));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_PMaxOA=0xe408, SETTING, MT_CPAP, SESSION,
|
||||
"Prisma_PMaxOA", QObject::tr("PMaxOA"),
|
||||
QObject::tr("PMaxOA"),
|
||||
QObject::tr("PMaxOA"),
|
||||
STR_UNIT_CMH2O, LOOKUP, Qt::green));
|
||||
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_ObstructLevel=0xe440, WAVEFORM, MT_CPAP, SESSION,
|
||||
"Prisma_ObstructLevel",
|
||||
QObject::tr("ObstructLevel"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("Obstruction Level"),
|
||||
QObject::tr("ObstructLevel"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("light purple")));
|
||||
chan->setUpperThreshold(100);
|
||||
chan->setLowerThreshold(0);
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_rMVFluctuation=0xe441, WAVEFORM, MT_CPAP, SESSION,
|
||||
"Prisma_rMVFluctuation",
|
||||
QObject::tr("rMVFluctuation"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("rMVFluctuation"),
|
||||
QObject::tr("rMVFluctuation"),
|
||||
STR_UNIT_Unknown, DEFAULT, QColor("light purple")));
|
||||
chan->setUpperThreshold(16);
|
||||
chan->setLowerThreshold(0);
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_rRMV=0xe442, WAVEFORM, MT_CPAP, SESSION,
|
||||
"Prisma_rRMV",
|
||||
QObject::tr("rRMV"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("rRMV"),
|
||||
QObject::tr("rRMV"),
|
||||
STR_UNIT_Unknown, DEFAULT, QColor("light purple")));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_PressureMeasured=0xe443, WAVEFORM, MT_CPAP, SESSION,
|
||||
"Prisma_PressureMeasured",
|
||||
QObject::tr("PressureMeasured"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("PressureMeasured"),
|
||||
QObject::tr("PressureMeasured"),
|
||||
STR_UNIT_CMH2O, DEFAULT, QColor("black")));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_FlowFull=0xe444, WAVEFORM, MT_CPAP, SESSION,
|
||||
"Prisma_FlowFull",
|
||||
QObject::tr("FlowFull"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("FlowFull"),
|
||||
QObject::tr("FlowFull"),
|
||||
STR_UNIT_Unknown, DEFAULT, QColor("black")));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_SPRStatus=0xe445, WAVEFORM, MT_CPAP, SESSION,
|
||||
"Prisma_SPRStatus",
|
||||
QObject::tr("SPRStatus"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("SPRStatus"),
|
||||
QObject::tr("SPRStatus"),
|
||||
STR_UNIT_Unknown, DEFAULT, QColor("black")));
|
||||
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_Artifact=0xe446, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_Artifact",
|
||||
QObject::tr("Artifact"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("Artifact"),
|
||||
QObject::tr("ART"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("salmon")));
|
||||
|
||||
channel.add(GRP_CPAP, new Channel(Prisma_CriticalLeak = 0xe447, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_CriticalLeak",
|
||||
QObject::tr("CriticalLeak"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("CriticalLeak"),
|
||||
QObject::tr("CL"),
|
||||
STR_UNIT_EventsPerHour, DEFAULT, QColor("orchid")));
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_eMO = 0xe448, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_eMO",
|
||||
QObject::tr("eMO"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("eMO"),
|
||||
QObject::tr("eMO"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("red")));
|
||||
chan->setEnabled(false);
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_eSO = 0xe449, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_eSO",
|
||||
QObject::tr("eSO"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("eSO"),
|
||||
QObject::tr("eSO"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("orange")));
|
||||
chan->setEnabled(false);
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_eS = 0xe44a, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_eS",
|
||||
QObject::tr("eS"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("eS"),
|
||||
QObject::tr("eS"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("light green")));
|
||||
chan->setEnabled(false);
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_eF = 0xe44b, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_eFL",
|
||||
QObject::tr("eFL"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("eFL"),
|
||||
QObject::tr("eFL"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("yellow")));
|
||||
chan->setEnabled(false);
|
||||
|
||||
channel.add(GRP_CPAP, chan = new Channel(Prisma_DeepSleep = 0xe44c, SPAN, MT_CPAP, SESSION,
|
||||
"Prisma_DS",
|
||||
QObject::tr("DeepSleep"),
|
||||
// TODO AXT add desc
|
||||
QObject::tr("DeepSleep"),
|
||||
QObject::tr("DS"),
|
||||
STR_UNIT_Percentage, DEFAULT, QColor("light blue")));
|
||||
chan->setEnabled(false);
|
||||
|
||||
}
|
||||
|
||||
bool PrismaLoader::initialized = false;
|
||||
|
||||
void PrismaLoader::Register()
|
||||
{
|
||||
if (initialized) { return; }
|
||||
|
||||
qDebug() << "Registering PrismaLoader";
|
||||
RegisterLoader(new PrismaLoader());
|
||||
initialized = true;
|
||||
}
|
235
oscar/SleepLib/loader_plugins/prisma_loader.h
Normal file
235
oscar/SleepLib/loader_plugins/prisma_loader.h
Normal file
@ -0,0 +1,235 @@
|
||||
/* SleepLib Löwenstein Prisma Loader Header
|
||||
*
|
||||
* Copyright (c) 2019-2022 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 PRISMA_LOADER_H
|
||||
#define PRISMA_LOADER_H
|
||||
#include "SleepLib/machine_loader.h"
|
||||
#include "SleepLib/loader_plugins/edfparser.h"
|
||||
|
||||
#ifdef UNITTEST_MODE
|
||||
#define private public
|
||||
#define protected public
|
||||
#endif
|
||||
|
||||
//********************************************************************************************
|
||||
/// IMPORTANT!!!
|
||||
//********************************************************************************************
|
||||
// Please INCREMENT the following value when making changes to this loaders implementation
|
||||
// BEFORE making a release
|
||||
const int prisma_data_version = 1;
|
||||
//
|
||||
//********************************************************************************************
|
||||
const QString prisma_class_name = STR_MACH_Prisma;
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
enum Prisma_Parameters {
|
||||
PRISMA_MODE = 6,
|
||||
PRISMA_PRESSURE = 9,
|
||||
PRISMA_PRESSURE_MAX = 10,
|
||||
PRISMA_PSOFT_MIN = 11,
|
||||
PRISMA_PSOFT = 12,
|
||||
PRISMA_SOFTPAP = 13,
|
||||
PRISMA_APAP_DYNAMIC = 15,
|
||||
PRISMA_HUMIDLEVEL = 16,
|
||||
PRISMA_AUTOSTART = 17,
|
||||
PRISMA_SOFTSTART_TIME_MAX = 18,
|
||||
PRISMA_SOFTSTART_TIME = 19,
|
||||
PRISMA_TUBE_TYPE = 21,
|
||||
PRISMA_PMAXOA = 38
|
||||
};
|
||||
|
||||
enum Prisma_Mode {
|
||||
PRISMA_MODE_CPAP = 1,
|
||||
PRISMA_MODE_APAP = 2,
|
||||
};
|
||||
|
||||
enum Prisma_APAP_Mode {
|
||||
PRISMA_APAP_MODE_STANDARD = 1,
|
||||
PRISMA_APAP_MODE_DYNAMIC = 2,
|
||||
};
|
||||
|
||||
enum Prisma_SoftPAP_Mode {
|
||||
Prisma_SoftPAP_OFF = 0,
|
||||
Prisma_SoftPAP_SLIGHT = 1,
|
||||
Prisma_SoftPAP_STANDARD = 2
|
||||
};
|
||||
|
||||
enum Prisma_Combined_Mode {
|
||||
PRISMA_COMBINED_MODE_CPAP = 1,
|
||||
PRISMA_COMBINED_MODE_APAP_STD = 2,
|
||||
PRISMA_COMBINED_MODE_APAP_DYN = 3,
|
||||
};
|
||||
|
||||
enum Prisma_Event_Type {
|
||||
PRISMA_EVENT_EPOCH_SEVERE_OBSTRUCTION = 1,
|
||||
PRISMA_EVENT_EPOCH_MILD_OBSTRUCTION = 2,
|
||||
PRISMA_EVENT_EPOCH_FLOW_LIMITATION = 3,
|
||||
PRISMA_EVENT_EPOCH_SNORE = 4,
|
||||
PRISMA_EVENT_EPOCH_PERIODIC_BREATHING = 5,
|
||||
PRISMA_EVENT_OBSTRUCTIVE_APNEA = 101,
|
||||
PRISMA_EVENT_CENTRAL_APNEA = 102,
|
||||
PRISMA_EVENT_APNEA_LEAKAGE = 103,
|
||||
PRISMA_EVENT_APNEA_HIGH_PRESSURE = 105,
|
||||
PRISMA_EVENT_APNEA_MOVEMENT = 106,
|
||||
PRISMA_EVENT_OBSTRUCTIVE_HYPOPNEA= 111,
|
||||
PRISMA_EVENT_CENTRAL_HYPOPNEA = 112,
|
||||
PRISMA_EVENT_HYPOPNEA_LEAKAGE = 113,
|
||||
PRISMA_EVENT_RERA = 121,
|
||||
PRISMA_EVENT_SNORE = 131,
|
||||
PRISMA_EVENT_ARTIFACT = 141,
|
||||
PRISMA_EVENT_FLOW_LIMITATION = 151,
|
||||
PRISMA_EVENT_CRITICAL_LEAKAGE = 161,
|
||||
PRISMA_EVENT_CS_RESPIRATION = 181,
|
||||
PRISMA_EVENT_EPOCH_DEEPSLEEP = 261,
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
class WMEDFInfo : public EDFInfo {
|
||||
virtual bool ParseSignalData();
|
||||
|
||||
protected:
|
||||
qint8 Read8S();
|
||||
quint8 Read8U();
|
||||
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
class PrismaLoader;
|
||||
class PrismaEventFile;
|
||||
|
||||
/*! \class PrismaImport
|
||||
* \brief Contains the functions to parse a single session... multithreaded */
|
||||
class PrismaImport:public ImportTask
|
||||
{
|
||||
public:
|
||||
PrismaImport(PrismaLoader * l, const MachineInfo& m, SessionID s, QString e, QString d): loader(l), machineInfo(m), sessionid(s), eventFileName(e), signalFileName(d) {}
|
||||
virtual ~PrismaImport() {};
|
||||
|
||||
//! \brief PrismaImport thread starts execution here.
|
||||
virtual void run();
|
||||
|
||||
protected:
|
||||
PrismaLoader * loader;
|
||||
const MachineInfo & machineInfo;
|
||||
SessionID sessionid;
|
||||
QString eventFileName;
|
||||
QString signalFileName;
|
||||
qint64 startdate;
|
||||
qint64 enddate;
|
||||
WMEDFInfo wmedf;
|
||||
PrismaEventFile * eventFile;
|
||||
Session * session;
|
||||
|
||||
void AddWaveform(ChannelID code, QString edfLabel);
|
||||
void AddEvents(ChannelID channel, Prisma_Event_Type eventType) {
|
||||
QList<Prisma_Event_Type> eventTypes = { eventType };
|
||||
AddEvents(channel, eventTypes);
|
||||
}
|
||||
void AddEvents(ChannelID channel, QList<Prisma_Event_Type> eventTypes);
|
||||
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
/*! \class PrismaLoader
|
||||
\brief Löwenstein Prisma Loader Module
|
||||
*/
|
||||
class PrismaLoader : public CPAPLoader
|
||||
{
|
||||
Q_OBJECT
|
||||
static bool initialized;
|
||||
public:
|
||||
PrismaLoader();
|
||||
virtual ~PrismaLoader();
|
||||
|
||||
//! \brief Detect if the given path contains a valid Folder structure
|
||||
virtual bool Detect(const QString & path);
|
||||
|
||||
//! \brief Load MachineInfo structure.
|
||||
virtual MachineInfo PeekInfo(const QString & path);
|
||||
|
||||
//! \brief Scans directory path for valid Prisma signature
|
||||
virtual int Open(const QString & path);
|
||||
|
||||
//! \brief Returns the database version of this loader
|
||||
virtual int Version() { return prisma_data_version; }
|
||||
|
||||
//! \brief Return the loaderName, in this case "Prisma"
|
||||
virtual const QString &loaderName() { return prisma_class_name; }
|
||||
|
||||
//! \brief Register this Module to the list of Loaders, so it knows to search for Prisma data.
|
||||
static void Register();
|
||||
|
||||
//! \brief Generate a generic MachineInfo structure, with basic Prisma info to be expanded upon.
|
||||
virtual MachineInfo newInfo() {
|
||||
return MachineInfo(MT_CPAP, 0, prisma_class_name, QObject::tr("Löwenstein"), QObject::tr("Prisma Smart"), QString(), QString(), QObject::tr(""), QDateTime::currentDateTime(), prisma_data_version);
|
||||
}
|
||||
|
||||
virtual QString PresReliefLabel();
|
||||
virtual ChannelID CPAPModeChannel();
|
||||
virtual ChannelID PresReliefMode();
|
||||
|
||||
|
||||
//! \brief Called at application init, to set up any custom Prisma Channels
|
||||
virtual void initChannels();
|
||||
|
||||
QHash<SessionID, PrismaImport*> sesstasks;
|
||||
|
||||
protected:
|
||||
|
||||
MachineInfo PeekInfoFromConfig(const QString & configPath);
|
||||
|
||||
//! \brief Scans the given directories for session data and create an import task for each logical session.
|
||||
void ScanFiles(const MachineInfo& info, const QString & path);
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
class PrismaEvent
|
||||
{
|
||||
public:
|
||||
PrismaEvent(int endTime, int duration, int pressure, int strength) : m_endTime(endTime), m_duration(duration), m_pressure(pressure), m_strenght(strength) {}
|
||||
int endTime() { return m_endTime; }
|
||||
int duration() { return m_duration; }
|
||||
int strength() { return m_strenght; }
|
||||
protected:
|
||||
int m_endTime;
|
||||
int m_duration;
|
||||
int m_pressure;
|
||||
int m_strenght;
|
||||
};
|
||||
|
||||
class PrismaEventFile
|
||||
{
|
||||
public:
|
||||
PrismaEventFile(QString fname);
|
||||
QHash<int, int> getParameters() {return m_parameters; }
|
||||
QList<PrismaEvent> getEvents(int eventId) {return m_events.contains(eventId) ? m_events[eventId] : QList<PrismaEvent>(); }
|
||||
|
||||
protected:
|
||||
QHash<int, int> m_parameters;
|
||||
QHash<int, QList<PrismaEvent>> m_events;
|
||||
};
|
||||
|
||||
//********************************************************************************************
|
||||
|
||||
class PrismaModelInfo
|
||||
{
|
||||
protected:
|
||||
QHash<QString,const char*> m_modelNames;
|
||||
|
||||
public:
|
||||
PrismaModelInfo();
|
||||
bool IsTested(const QString & deviceId) const;
|
||||
const char* Name(const QString & deviceId) const;
|
||||
};
|
||||
|
||||
#endif // PRISMA_LOADER_H
|
@ -46,6 +46,7 @@
|
||||
#include "SleepLib/loader_plugins/sleepstyle_loader.h"
|
||||
#include "SleepLib/loader_plugins/weinmann_loader.h"
|
||||
#include "SleepLib/loader_plugins/viatom_loader.h"
|
||||
#include "SleepLib/loader_plugins/prisma_loader.h"
|
||||
|
||||
MainWindow *mainwin = nullptr;
|
||||
|
||||
@ -691,6 +692,7 @@ int main(int argc, char *argv[]) {
|
||||
CMS50F37Loader::Register();
|
||||
MD300W1Loader::Register();
|
||||
ViatomLoader::Register();
|
||||
PrismaLoader::Register();
|
||||
|
||||
// Begin logging device connection activity.
|
||||
QString connectionsLogDir = GetLogDir() + "/connections";
|
||||
|
@ -301,6 +301,7 @@ SOURCES += \
|
||||
SleepLib/loader_plugins/sleepstyle_EDFinfo.cpp \
|
||||
SleepLib/loader_plugins/intellipap_loader.cpp \
|
||||
SleepLib/loader_plugins/mseries_loader.cpp \
|
||||
SleepLib/loader_plugins/prisma_loader.cpp \
|
||||
SleepLib/loader_plugins/prs1_loader.cpp \
|
||||
SleepLib/loader_plugins/prs1_parser.cpp \
|
||||
SleepLib/loader_plugins/prs1_parser_xpap.cpp \
|
||||
@ -403,6 +404,7 @@ HEADERS += \
|
||||
SleepLib/loader_plugins/sleepstyle_EDFinfo.h \
|
||||
SleepLib/loader_plugins/intellipap_loader.h \
|
||||
SleepLib/loader_plugins/mseries_loader.h \
|
||||
SleepLib/loader_plugins/prisma_loader.h \
|
||||
SleepLib/loader_plugins/prs1_loader.h \
|
||||
SleepLib/loader_plugins/prs1_parser.h \
|
||||
SleepLib/loader_plugins/resmed_loader.h \
|
||||
|
Loading…
Reference in New Issue
Block a user