/* -*- 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 Machine Class Implementation * * Copyright (c) 2011-2014 Mark Watkins * * 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. */ #include #include #include #include #include #include #include #include "machine.h" #include "profiles.h" #include #include "SleepLib/schema.h" extern QProgressBar * qprogress; ////////////////////////////////////////////////////////////////////////////////////////// // Machine Base-Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// Machine::Machine(Profile *p,MachineID id) { day.clear(); highest_sessionid=0; profile=p; if (!id) { srand(time(NULL)); MachineID temp; do { temp = rand(); } while (profile->machlist.find(temp)!=profile->machlist.end()); m_id=temp; } else m_id=id; //qDebug() << "Create Machine: " << hex << m_id; //%lx",m_id); m_type=MT_UNKNOWN; firstsession=true; } Machine::~Machine() { qDebug() << "Destroy Machine" << m_class; for (QMap::iterator d=day.begin();d!=day.end();d++) { delete d.value(); } } Session *Machine::SessionExists(SessionID session) { if (sessionlist.find(session)!=sessionlist.end()) { return sessionlist[session]; } else { return NULL; } } // Find date this session belongs in QDate Machine::pickDate(qint64 first) { QTime split_time=PROFILE.session->daySplitTime(); int combine_sessions=PROFILE.session->combineCloseSessions(); QDateTime d2=QDateTime::fromTime_t(first/1000); QDate date=d2.date(); QTime time=d2.time(); int closest_session=0; if (time 0) { QMap::iterator dit=day.find(date.addDays(-1)); // Check Day Before if (dit != day.end()) { QDateTime lt=QDateTime::fromTime_t(dit.value()->last()/1000L); closest_session=lt.secsTo(d2)/60; if (closest_session < combine_sessions) { date=date.addDays(-1); } } } return date; } QDate Machine::AddSession(Session *s,Profile *p) { if (!s) { qWarning() << "Empty Session in Machine::AddSession()"; return QDate(); } if (!p) { qWarning() << "Empty Profile in Machine::AddSession()"; return QDate(); } if (s->session()>highest_sessionid) highest_sessionid=s->session(); QTime split_time=PROFILE.session->daySplitTime(); int combine_sessions=PROFILE.session->combineCloseSessions(); int ignore_sessions=PROFILE.session->ignoreShortSessions(); int session_length=s->last()-s->first(); session_length/=60000; sessionlist[s->session()]=s; // To make sure it get's saved later even if it's not wanted. //int drift=PROFILE.cpap->clockDrift(); QDateTime d2=QDateTime::fromTime_t(s->first()/1000); QDate date=d2.date(); QTime time=d2.time(); QMap::iterator dit,nextday; bool combine_next_day=false; int closest_session=0; if (time 0) { dit=day.find(date.addDays(-1)); // Check Day Before if (dit!=day.end()) { QDateTime lt=QDateTime::fromTime_t(dit.value()->last()/1000); closest_session=lt.secsTo(d2)/60; if (closest_sessionfirst()/1000); closest_session=d2.secsTo(lt)/60; if (closest_session < combine_sessions) { // add todays here. pull all tomorrows records to this date. combine_next_day=true; } } } } if (session_length=60)) return QDate(); } if (!firstsession) { if (firstday>date) firstday=date; if (lastdayAddDay(date,dd,m_type); } else { dd=*dit; } dd->AddSession(s); if (combine_next_day) { for (QList::iterator i=nextday.value()->begin();i!=nextday.value()->end();i++) { dd->AddSession(*i); } QMap >::iterator nd=p->daylist.find(date.addDays(1)); for (QList::iterator i=nd->begin();i!=nd->end();i++) { if (*i==nextday.value()) { nd.value().erase(i); } } day.erase(nextday); } return date; } // This functions purpose is murder and mayhem... It deletes all of a machines data. // Therefore this is the most dangerous function in this software.. bool Machine::Purge(int secret) { // Boring api key to stop this function getting called by accident :) if (secret!=3478216) return false; // It would be joyous if this function screwed up.. QString path=profile->Get(properties[STR_PROP_Path]); //STR_GEN_DataFolder)+"/"+m_class+"_"; //if (properties.contains(STR_PROP_Serial)) path+=properties[STR_PROP_Serial]; else path+=hexid(); QDir dir(path); if (!dir.exists()) // It doesn't exist anyway. return true; if (!dir.isReadable()) return false; qDebug() << "Purging " << QDir::toNativeSeparators(path); dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); dir.setSorting(QDir::Name); QFileInfoList list=dir.entryInfoList(); int could_not_kill=0; for (int i=0;i0) { // qWarning() << "Could not purge path\n" << path << "\n\n" << could_not_kill << " file(s) remain.. Suggest manually deleting this path\n"; // return false; } return true; } const quint32 channel_version=1; bool Machine::Load() { QString path=profile->Get(properties[STR_PROP_Path]); //STR_GEN_DataFolder)+"/"+m_class+"_"+hexid(); QDir dir(path); qDebug() << "Loading " << QDir::toNativeSeparators(path); if (!dir.exists() || !dir.isReadable()) return false; dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); dir.setSorting(QDir::Name); QFileInfoList list=dir.entryInfoList(); typedef QVector StringList; QMap sessfiles; QMap::iterator s; QString fullpath,ext_s,sesstr; int ext; SessionID sessid; bool ok; for (int i=0;isetValue((float(cnt)/float(size)*100.0)); QApplication::processEvents(); } Session *sess=new Session(this,s.key()); if (sess->LoadSummary(s.value()[0])) { sess->SetEventFile(s.value()[1]); //sess->OpenEvents(); AddSession(sess,profile); } else { qWarning() << "Error unpacking summary data"; delete sess; } } if (qprogress) qprogress->setValue(100); return true; } bool Machine::SaveSession(Session *sess) { QString path=profile->Get(properties[STR_PROP_Path]); //STR_GEN_DataFolder)+"/"+m_class+"_"+hexid(); if (sess->IsChanged()) sess->Store(path); return true; } bool Machine::Save() { //int size; int cnt=0; QString path=profile->Get(properties[STR_PROP_Path]); //STR_GEN_DataFolder)+"/"+m_class+"_"+hexid(); QDir dir(path); if (!dir.exists()) { dir.mkdir(path); } QHash::iterator s; m_savelist.clear(); for (s=sessionlist.begin(); s!=sessionlist.end(); s++) { cnt++; if ((*s)->IsChanged()) { m_savelist.push_back(*s); } } savelistCnt=0; savelistSize=m_savelist.size(); bool cachesessions=PROFILE.session->cacheSessions(); if (!PROFILE.session->multithreading()) { for (int i=0;isetValue(0+(float(savelistCnt)/float(savelistSize)*100.0)); QApplication::processEvents(); } Session *s=m_savelist.at(i); s->UpdateSummaries(); s->Store(path); if (!cachesessions) s->TrashEvents(); savelistCnt++; } return true; } int threads=QThread::idealThreadCount(); savelistSem=new QSemaphore(threads); savelistSem->acquire(threads); QVectorthread; for (int i=0;istart(); } while (!savelistSem->tryAcquire(threads,250)) { if (qprogress) { // qprogress->setValue(66.0+(float(savelistCnt)/float(savelistSize)*33.0)); QApplication::processEvents(); } } for (int i=0;iisRunning()) { SaveThread::msleep(250); QApplication::processEvents(); } delete thread[i]; } delete savelistSem; return true; } /*SaveThread::SaveThread(Machine *m,QString p) { machine=m; path=p; } */ void SaveThread::run() { bool cachesessions=PROFILE.session->cacheSessions(); while (Session *sess=machine->popSaveList()) { int i=(float(machine->savelistCnt)/float(machine->savelistSize)*100.0); emit UpdateProgress(i); sess->UpdateSummaries(); sess->Store(path); if (!cachesessions) sess->TrashEvents(); } machine->savelistSem->release(1); } Session *Machine::popSaveList() { Session *sess=NULL; savelistMutex.lock(); if (m_savelist.size()>0) { sess=m_savelist.at(0); m_savelist.pop_front(); savelistCnt++; } savelistMutex.unlock(); return sess; } ////////////////////////////////////////////////////////////////////////////////////////// // CPAP implmementation ////////////////////////////////////////////////////////////////////////////////////////// CPAP::CPAP(Profile *p,MachineID id):Machine(p,id) { m_type=MT_CPAP; } CPAP::~CPAP() { } ////////////////////////////////////////////////////////////////////////////////////////// // Oximeter Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// Oximeter::Oximeter(Profile *p,MachineID id):Machine(p,id) { m_type=MT_OXIMETER; } Oximeter::~Oximeter() { } ////////////////////////////////////////////////////////////////////////////////////////// // SleepStage Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// SleepStage::SleepStage(Profile *p,MachineID id):Machine(p,id) { m_type=MT_SLEEPSTAGE; } SleepStage::~SleepStage() { } ChannelID NoChannel, SESSION_ENABLED; ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_Pressure, CPAP_PS, CPAP_Mode, CPAP_AHI, CPAP_PressureMin, CPAP_PressureMax, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive, CPAP_Hypopnea, CPAP_ClearAirway, CPAP_Apnea, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, CPAP_VSnore2, CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_FlowRate, CPAP_MaskPressure, CPAP_MaskPressureHi, CPAP_RespEvent, CPAP_Snore, CPAP_MinuteVent, CPAP_RespRate, CPAP_TidalVolume, CPAP_PTB, CPAP_Leak, CPAP_LeakMedian, CPAP_LeakTotal, CPAP_MaxLeak, CPAP_FLG, CPAP_IE, CPAP_Te, CPAP_Ti, CPAP_TgMV, CPAP_UserFlag1, CPAP_UserFlag2, CPAP_UserFlag3, CPAP_BrokenSummary, CPAP_BrokenWaveform, CPAP_RDI, CPAP_PresReliefSet, CPAP_PresReliefMode, CPAP_PresReliefType, CPAP_PSMin, CPAP_PSMax, CPAP_Test1, CPAP_Test2; ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure; ChannelID INTP_SmartFlex; ChannelID INTELLIPAP_Unknown1, INTELLIPAP_Unknown2; ChannelID PRS1_00, PRS1_01, PRS1_08, PRS1_0A, PRS1_0B, PRS1_0C, PRS1_0E, PRS1_0F, PRS1_10, PRS1_12, PRS1_FlexMode, PRS1_FlexSet, PRS1_HumidStatus, CPAP_HumidSetting, PRS1_SysLock, PRS1_SysOneResistStat, PRS1_SysOneResistSet, PRS1_HoseDiam, PRS1_AutoOn, PRS1_AutoOff, PRS1_MaskAlert, PRS1_ShowAHI; ChannelID OXI_Pulse, OXI_SPO2, OXI_PulseChange, OXI_SPO2Drop, OXI_Plethy; ChannelID Journal_Notes, Journal_Weight, Journal_BMI, Journal_ZombieMeter, Bookmark_Start, Bookmark_End, Bookmark_Notes; ChannelID ZEO_SleepStage, ZEO_ZQ, ZEO_TotalZ, ZEO_TimeToZ, ZEO_TimeInWake, ZEO_TimeInREM, ZEO_TimeInLight, ZEO_TimeInDeep, ZEO_Awakenings, ZEO_AlarmReason, ZEO_SnoozeTime, ZEO_WakeTone, ZEO_WakeWindow, ZEO_AlarmType, ZEO_MorningFeel, ZEO_FirmwareVersion, ZEO_FirstAlarmRing, ZEO_LastAlarmRing, ZEO_FirstSnoozeTime, ZEO_LastSnoozeTime, ZEO_SetAlarmTime, ZEO_RiseTime;