mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
1423 lines
40 KiB
C++
1423 lines
40 KiB
C++
#include <tr1/random>
|
|
#include <wx/colour.h>
|
|
#include <wx/log.h>
|
|
#include <wx/progdlg.h>
|
|
#include "binary_file.h"
|
|
#include "machine.h"
|
|
#include "profiles.h"
|
|
#include <algorithm>
|
|
#include <initializer_list>
|
|
|
|
extern wxProgressDialog *loader_progress;
|
|
|
|
map<MachineType,ChannelCode> MachLastCode;
|
|
|
|
|
|
ChannelCode RegisterChannel(MachineType type)
|
|
{
|
|
if (MachLastCode.find(type)==MachLastCode.end()) {
|
|
return MachLastCode[type]=0;
|
|
}
|
|
return ++(MachLastCode[type]);
|
|
};
|
|
|
|
const int CPLAP_FLONG=RegisterChannel(MT_CPAP);
|
|
//map<MachineID,Machine *> MachList;
|
|
|
|
/*map<MachineType,wxString> MachineTypeString= {
|
|
{MT_UNKNOWN, wxT("Unknown")},
|
|
{MT_CPAP, wxT("CPAP")},
|
|
{MT_OXIMETER, wxT("Oximeter")},
|
|
{MT_SLEEPSTAGE, wxT("SleepStage")}
|
|
}; */
|
|
|
|
/*map<wxString,MachineType> MachineTypeLookup= {
|
|
{ MachineTypeString[MT_UNKNOWN].Lower(), MT_UNKNOWN },
|
|
{ MachineTypeString[MT_CPAP].Lower(), MT_CPAP },
|
|
{ MachineTypeString[MT_OXIMETER].Lower(), MT_OXIMETER },
|
|
{ MachineTypeString[MT_SLEEPSTAGE].Lower(), MT_SLEEPSTAGE}
|
|
}; */
|
|
map<CPAPMode,wxString> CPAPModeNames;
|
|
/*={
|
|
{MODE_UNKNOWN,wxT("Undetermined")},
|
|
{MODE_CPAP,wxT("CPAP")},
|
|
{MODE_APAP,wxT("APAP")},
|
|
{MODE_BIPAP,wxT("BIPAP")},
|
|
{MODE_ASV,wxT("ASV")}
|
|
};*/
|
|
map<PRTypes,wxString> PressureReliefNames;
|
|
/*={
|
|
{PR_UNKNOWN,_("Undetermined")},
|
|
{PR_NONE,_("None")},
|
|
{PR_CFLEX,wxT("C-Flex")},
|
|
{PR_CFLEXPLUS,wxT("C-Flex+")},
|
|
{PR_AFLEX,wxT("A-Flex")},
|
|
{PR_BIFLEX,wxT("Bi-Flex")},
|
|
{PR_EPR,wxT("Exhalation Pressure Relief (EPR)")},
|
|
{PR_SMARTFLEX,wxT("SmartFlex")}
|
|
}; */
|
|
|
|
|
|
|
|
// Master list. Look up local name table first.. then these if not found.
|
|
map<MachineCode,wxString> DefaultMCShortNames;
|
|
/*= {
|
|
{CPAP_Obstructive, wxT("OA")},
|
|
{CPAP_Hypopnea, wxT("H")},
|
|
{CPAP_RERA, wxT("RE")},
|
|
{CPAP_ClearAirway, wxT("CA")},
|
|
{CPAP_CSR, wxT("CSR")},
|
|
{CPAP_VSnore, wxT("VS")},
|
|
{CPAP_FlowLimit, wxT("FL")},
|
|
{CPAP_Pressure, wxT("P")},
|
|
{CPAP_Leak, wxT("LR")},
|
|
{CPAP_EAP, wxT("EAP")},
|
|
{CPAP_IAP, wxT("IAP")},
|
|
{PRS1_VSnore2, wxT("VS")},
|
|
{PRS1_PressurePulse,wxT("PP")}
|
|
};
|
|
|
|
// Master list. Look up local name table first.. then these if not found.
|
|
map<MachineCode,wxString> DefaultMCLongNames= {
|
|
{CPAP_Obstructive, wxT("Obstructive Apnea")},
|
|
{CPAP_Hypopnea, wxT("Hypopnea")},
|
|
{CPAP_RERA, wxT("Respiratory Effort / Arrousal")},
|
|
{CPAP_ClearAirway, wxT("Clear Airway Apnea")},
|
|
{CPAP_CSR, wxT("Cheyne Stokes Respiration")},
|
|
{CPAP_VSnore, wxT("Vibratory Snore")},
|
|
{CPAP_FlowLimit, wxT("Flow Limitation")},
|
|
{CPAP_Pressure, wxT("Pressure")},
|
|
{CPAP_Leak, wxT("Leak Rate")},
|
|
{CPAP_EAP, wxT("BIPAP Exhalation Pressure")},
|
|
{CPAP_IAP, wxT("BIPAP Inhalation Pressure")},
|
|
{PRS1_VSnore2, wxT("Vibratory Snore")},
|
|
{PRS1_PressurePulse,wxT("Pressue Pulse")}
|
|
}; */
|
|
|
|
void InitMapsWithoutAwesomeInitializerLists()
|
|
{
|
|
CPAPModeNames[MODE_UNKNOWN]=wxT("Undetermined");
|
|
CPAPModeNames[MODE_CPAP]=wxT("CPAP");
|
|
CPAPModeNames[MODE_APAP]=wxT("APAP");
|
|
CPAPModeNames[MODE_BIPAP]=wxT("BIPAP");
|
|
CPAPModeNames[MODE_ASV]=wxT("ASV");
|
|
|
|
PressureReliefNames[PR_UNKNOWN]=_("Undetermined");
|
|
PressureReliefNames[PR_NONE]=_("None");
|
|
PressureReliefNames[PR_CFLEX]=wxT("C-Flex");
|
|
PressureReliefNames[PR_CFLEXPLUS]=wxT("C-Flex+");
|
|
PressureReliefNames[PR_AFLEX]=wxT("A-Flex");
|
|
PressureReliefNames[PR_BIFLEX]=wxT("Bi-Flex");
|
|
PressureReliefNames[PR_EPR]=wxT("Exhalation Pressure Relief (EPR)");
|
|
PressureReliefNames[PR_SMARTFLEX]=wxT("SmartFlex");
|
|
|
|
DefaultMCShortNames[CPAP_Obstructive]=wxT("OA");
|
|
DefaultMCShortNames[CPAP_Hypopnea]=wxT("H");
|
|
DefaultMCShortNames[CPAP_RERA]=wxT("RE");
|
|
DefaultMCShortNames[CPAP_ClearAirway]=wxT("CA");
|
|
DefaultMCShortNames[CPAP_CSR]=wxT("CSR");
|
|
DefaultMCShortNames[CPAP_VSnore]=wxT("VS");
|
|
DefaultMCShortNames[CPAP_FlowLimit]=wxT("FL");
|
|
DefaultMCShortNames[CPAP_Pressure]=wxT("P");
|
|
DefaultMCShortNames[CPAP_Leak]=wxT("LR");
|
|
DefaultMCShortNames[CPAP_EAP]=wxT("EAP");
|
|
DefaultMCShortNames[CPAP_IAP]=wxT("IAP");
|
|
DefaultMCShortNames[PRS1_VSnore2]=wxT("VS");
|
|
DefaultMCShortNames[PRS1_PressurePulse]=wxT("PP");
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is technically gui related.. however I have something in mind for it.
|
|
/*const map<MachineCode,FlagType> DefaultFlagTypes= {
|
|
{CPAP_Obstructive, FT_BAR},
|
|
{CPAP_Hypopnea, FT_BAR},
|
|
{CPAP_RERA, FT_BAR},
|
|
{CPAP_VSnore, FT_BAR},
|
|
{PRS1_VSnore2, FT_BAR},
|
|
{CPAP_FlowLimit, FT_BAR},
|
|
{CPAP_ClearAirway, FT_BAR},
|
|
{CPAP_CSR, FT_SPAN},
|
|
{PRS1_PressurePulse,FT_DOT},
|
|
{CPAP_Pressure, FT_DOT}
|
|
};
|
|
|
|
const unsigned char flagalpha=0x80;
|
|
const map<MachineCode,wxColour> DefaultFlagColours= {
|
|
{CPAP_Obstructive, wxColour(0x80,0x80,0xff,flagalpha)},
|
|
{CPAP_Hypopnea, wxColour(0x00,0x00,0xff,flagalpha)},
|
|
{CPAP_RERA, wxColour(0x40,0x80,0xff,flagalpha)},
|
|
{CPAP_VSnore, wxColour(0xff,0x20,0x20,flagalpha)},
|
|
{CPAP_FlowLimit, wxColour(0x20,0x20,0x20,flagalpha)},
|
|
{CPAP_ClearAirway, wxColour(0xff,0x40,0xff,flagalpha)},
|
|
{CPAP_CSR, wxColour(0x40,0xff,0x40,flagalpha)},
|
|
{PRS1_VSnore2, wxColour(0xff,0x20,0x20,flagalpha)},
|
|
{PRS1_PressurePulse,wxColour(0xff,0x40,0xff,flagalpha)}
|
|
}; */
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// Machine Base-Class implmementation
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
Machine::Machine(Profile *p,MachineID id)
|
|
{
|
|
wxLogDebug(wxT("Create Machine"));
|
|
profile=p;
|
|
if (!id) {
|
|
std::tr1::minstd_rand gen;
|
|
std::tr1::uniform_int<MachineID> unif(1, 0x7fffffff);
|
|
gen.seed((unsigned int) time(NULL));
|
|
MachineID temp;
|
|
do {
|
|
temp = unif(gen); //unif(gen) << 32 |
|
|
} while (profile->machlist.find(temp)!=profile->machlist.end());
|
|
|
|
m_id=temp;
|
|
|
|
} else m_id=id;
|
|
m_type=MT_UNKNOWN;
|
|
firstsession=true;
|
|
}
|
|
Machine::~Machine()
|
|
{
|
|
wxLogDebug(wxT("Destroy Machine"));
|
|
map<wxDateTime,Day *>::iterator d;
|
|
for (d=day.begin();d!=day.end();d++) {
|
|
delete d->second;
|
|
}
|
|
}
|
|
Session *Machine::SessionExists(SessionID session)
|
|
{
|
|
if (sessionlist.find(session)!=sessionlist.end()) {
|
|
return sessionlist[session];
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
Day *Machine::AddSession(Session *s,Profile *p)
|
|
{
|
|
wxTimeSpan span;
|
|
|
|
wxDateTime date=s->first()-wxTimeSpan::Day();
|
|
date.ResetTime();
|
|
|
|
const int hours_since_last_session=4;
|
|
const int hours_since_midnight=12;
|
|
|
|
bool previous=false;
|
|
|
|
// Find what day session belongs to.
|
|
if (day.find(date)!=day.end()) {
|
|
// Previous day record exists...
|
|
|
|
// Calculate time since end of previous days last session
|
|
span=s->first()-day[date]->last();
|
|
|
|
// less than n hours since last session yesterday?
|
|
if (span < wxTimeSpan::Hours(hours_since_last_session)) {
|
|
previous=true;
|
|
}
|
|
}
|
|
|
|
if (!previous) {
|
|
// Calculate Hours since midnight.
|
|
wxDateTime t=s->first();
|
|
t.ResetTime();
|
|
span=s->first()-t;
|
|
|
|
// Bedtime was late last night.
|
|
if (span < wxTimeSpan::Hours(hours_since_midnight)) {
|
|
previous=true;
|
|
}
|
|
}
|
|
|
|
if (!previous) {
|
|
// Revert to sessions original day.
|
|
date+=wxTimeSpan::Day();
|
|
}
|
|
|
|
sessionlist[s->session()]=s;
|
|
|
|
if (!firstsession) {
|
|
if (firstday>date) firstday=date;
|
|
if (lastday<date) lastday=date;
|
|
} else {
|
|
firstday=lastday=date;
|
|
firstsession=false;
|
|
}
|
|
if (day.find(date)==day.end()) {
|
|
day[date]=new Day(this);
|
|
p->AddDay(date,day[date],m_type);
|
|
}
|
|
|
|
day[date]->AddSession(s);
|
|
|
|
return day[date];
|
|
}
|
|
|
|
bool Machine::Load()
|
|
{
|
|
wxString path=profile->Get("DataFolder")+wxFileName::GetPathSeparator()+hexid();
|
|
wxDir dir;
|
|
wxLogMessage(wxT("Loading ")+path);
|
|
dir.Open(path);
|
|
if (!dir.IsOpened()) return false;
|
|
|
|
wxString filename;
|
|
bool cont=dir.GetFirst(&filename);
|
|
|
|
typedef vector<wxString> StringList;
|
|
map<SessionID,StringList> sessfiles;
|
|
map<SessionID,StringList>::iterator s;
|
|
while (cont) {
|
|
wxString ext_s=filename.AfterLast(wxChar('.'));
|
|
wxString session_s=wxT("0x")+filename.BeforeLast(wxChar('.'));
|
|
|
|
SessionID sessid;
|
|
long ext;
|
|
session_s.ToLong(&sessid,16);
|
|
ext_s.ToLong(&ext);
|
|
if (sessfiles[sessid].capacity()==0) sessfiles[sessid].resize(3);
|
|
wxString fullname=path+wxFileName::GetPathSeparator()+filename;
|
|
if (ext==0) sessfiles[sessid][0]=fullname;
|
|
else if (ext==1) sessfiles[sessid][1]=fullname;
|
|
else if (ext==2) sessfiles[sessid][2]=fullname;
|
|
|
|
cont=dir.GetNext(&filename);
|
|
}
|
|
|
|
for (s=sessfiles.begin(); s!=sessfiles.end(); s++) {
|
|
Session *sess=new Session(this,s->first);
|
|
if (sess->LoadSummary(s->second[0])) {
|
|
//sess->SetEventFile(sessfiles[sess->id()][1]);
|
|
//sess->SetWaveformFile(sessfiles[sess->id()][1]);
|
|
//wxString sx=sess->first().Format()+wxT(" ")+sess->last().Format();
|
|
// wxPrintf(s+wxT(" O=%i H=%i CA=%i \n"),sess->summary[CPAP_Obstructive].GetLong(),sess->summary[CPAP_Hypopnea].GetLong(),sess->summary[CPAP_ClearAirway].GetLong());
|
|
// sess->LoadEvents(s->second[1]);
|
|
// sess->LoadWaveforms(s->second[2]);
|
|
sess->SetEventFile(s->second[1]);
|
|
sess->SetWaveFile(s->second[2]);
|
|
|
|
AddSession(sess,profile);
|
|
} else {
|
|
delete sess;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
bool Machine::Save()
|
|
{
|
|
map<wxDateTime,Day *>::iterator d;
|
|
vector<Session *>::iterator s;
|
|
int size=0;
|
|
int cnt=0;
|
|
|
|
wxString path=profile->Get("DataFolder")+wxFileName::GetPathSeparator()+hexid();
|
|
|
|
// Calculate size for progress bar
|
|
for (d=day.begin();d!=day.end();d++)
|
|
size+=d->second->size();
|
|
|
|
for (d=day.begin();d!=day.end();d++) {
|
|
|
|
for (s=d->second->begin(); s!=d->second->end(); s++) {
|
|
cnt++;
|
|
if (loader_progress) loader_progress->Update(50+(float(cnt)/float(size)*50.0));
|
|
if ((*s)->IsChanged()) (*s)->Store(path);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// Day Class implmementation
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
Day::Day(Machine *m)
|
|
:machine(m)
|
|
{
|
|
|
|
d_firstsession=true;
|
|
sessions.clear();
|
|
}
|
|
Day::~Day()
|
|
{
|
|
vector<Session *>::iterator s;
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
delete (*s);
|
|
}
|
|
|
|
}
|
|
MachineType Day::machine_type()
|
|
{
|
|
return machine->GetType();
|
|
};
|
|
|
|
void Day::AddSession(Session *s)
|
|
{
|
|
if (!s) {
|
|
wxLogWarning(wxT("Day::AddSession called with NULL session object"));
|
|
return;
|
|
}
|
|
if (d_firstsession) {
|
|
d_firstsession=false;
|
|
d_first=s->first();
|
|
d_last=s->last();
|
|
} else {
|
|
if (d_first > s->first()) d_first = s->first();
|
|
if (d_last < s->last()) d_last = s->last();
|
|
}
|
|
sessions.push_back(s);
|
|
}
|
|
|
|
EventDataType Day::summary_sum(MachineCode code)
|
|
{
|
|
EventDataType val=0;
|
|
vector<Session *>::iterator s;
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.summary.find(code)!=sess.summary.end()) {
|
|
val+=sess.summary[code].GetDouble();
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
|
|
EventDataType Day::summary_max(MachineCode code)
|
|
{
|
|
EventDataType val=0,tmp;
|
|
bool fir=true;
|
|
// Cache this?
|
|
|
|
vector<Session *>::iterator s;
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.summary.find(code)!=sess.summary.end()) {
|
|
tmp=sess.summary[code].GetDouble();
|
|
if (tmp>val) val=tmp;
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
EventDataType Day::summary_min(MachineCode code)
|
|
{
|
|
EventDataType val=0,tmp;
|
|
bool fir=true;
|
|
// Cache this?
|
|
|
|
vector<Session *>::iterator s;
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.summary.find(code)!=sess.summary.end()) {
|
|
tmp=sess.summary[code].GetDouble();
|
|
if (fir) {
|
|
val=tmp;
|
|
fir=false;
|
|
} else {
|
|
if (val<tmp) val=tmp;
|
|
}
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
|
|
EventDataType Day::summary_avg(MachineCode code)
|
|
{
|
|
EventDataType val=0,tmp=0;
|
|
bool fir=true;
|
|
// Cache this?
|
|
|
|
int cnt=0;
|
|
vector<Session *>::iterator s;
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.summary.find(code)!=sess.summary.end()) {
|
|
tmp+=sess.summary[code].GetDouble();
|
|
cnt++;
|
|
}
|
|
}
|
|
val=tmp/EventDataType(cnt);
|
|
return val;
|
|
}
|
|
|
|
EventDataType Day::min(MachineCode code,int field)
|
|
{
|
|
EventDataType val=0,tmp;
|
|
bool fir=true;
|
|
// Cache this?
|
|
|
|
vector<Session *>::iterator s;
|
|
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
tmp=sess.min_event_field(code,field);
|
|
if (fir) {
|
|
val=tmp;
|
|
fir=false;
|
|
} else {
|
|
if (val>tmp) val=tmp;
|
|
}
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
|
|
EventDataType Day::max(MachineCode code,int field)
|
|
{
|
|
EventDataType val=0,tmp;
|
|
bool fir=true;
|
|
// Cache this?
|
|
|
|
// Don't assume sessions are in order.
|
|
vector<Session *>::iterator s;
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
tmp=sess.max_event_field(code,field);
|
|
if (fir) {
|
|
val=tmp;
|
|
fir=false;
|
|
} else {
|
|
if (val<tmp) val=tmp;
|
|
}
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
|
|
EventDataType Day::avg(MachineCode code,int field)
|
|
{
|
|
double val=0;
|
|
// Cache this?
|
|
int cnt=0;
|
|
vector<Session *>::iterator s;
|
|
|
|
// Don't assume sessions are in order.
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
val+=sess.avg_event_field(code,field);
|
|
cnt++;
|
|
}
|
|
}
|
|
if (cnt==0) cnt=1;
|
|
return EventDataType(val/cnt);
|
|
}
|
|
|
|
EventDataType Day::sum(MachineCode code,int field)
|
|
{
|
|
// Cache this?
|
|
EventDataType val=0;
|
|
vector<Session *>::iterator s;
|
|
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
val+=sess.sum_event_field(code,field);
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
|
|
EventDataType Day::count(MachineCode code)
|
|
{
|
|
EventDataType val=0;
|
|
// Cache this?
|
|
vector<Session *>::iterator s;
|
|
|
|
// Don't assume sessions are in order.
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
val+=sess.count_events(code);
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
EventDataType Day::weighted_avg(MachineCode code,int field)
|
|
{
|
|
double val=0;
|
|
// Cache this?
|
|
int cnt=0;
|
|
// Don't assume sessions are in order.
|
|
|
|
vector<Session *>::iterator s;
|
|
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
val+=sess.weighted_avg_event_field(code,field);
|
|
cnt++;
|
|
}
|
|
}
|
|
if (cnt==0) cnt=1;
|
|
return EventDataType(val/cnt);
|
|
}
|
|
wxTimeSpan Day::total_time()
|
|
{
|
|
//if (d_totaltime>wxTimeSpan::Seconds(0)) return d_totaltime;
|
|
|
|
vector<Session *>::iterator s;
|
|
|
|
d_totaltime=wxTimeSpan::Seconds(0);
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
d_totaltime+=sess.last()-sess.first();
|
|
if (d_totaltime>wxTimeSpan::Hours(15)) {
|
|
int c=1;
|
|
}
|
|
}
|
|
return d_totaltime;
|
|
}
|
|
EventDataType Day::percentile(MachineCode code,int field,double percent)
|
|
{
|
|
double val=0;
|
|
// Cache this?
|
|
int cnt=0;
|
|
vector<Session *>::iterator s;
|
|
|
|
// Don't assume sessions are in order.
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
val+=sess.percentile(code,field,percent);
|
|
cnt++;
|
|
}
|
|
}
|
|
if (cnt==0) return 0;
|
|
return EventDataType(val/cnt);
|
|
|
|
}
|
|
|
|
const wxDateTime & Day::first(MachineCode code)
|
|
{
|
|
static wxDateTime date;
|
|
wxDateTime tmp;
|
|
bool fir=true;
|
|
// Cache this?
|
|
vector<Session *>::iterator s;
|
|
|
|
// Don't assume sessions are in order.
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
tmp=sess.events[code][0]->time();
|
|
if (fir) {
|
|
date=tmp;
|
|
fir=false;
|
|
} else {
|
|
if (tmp<date) date=tmp;
|
|
}
|
|
}
|
|
}
|
|
return date;
|
|
}
|
|
|
|
const wxDateTime & Day::last(MachineCode code)
|
|
{
|
|
size_t cnt;
|
|
static wxDateTime date;
|
|
wxDateTime tmp;
|
|
bool fir=true;
|
|
// Cache this?
|
|
|
|
// Don't assume sessions are in order.
|
|
vector<Session *>::iterator s;
|
|
;
|
|
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
Session & sess=*(*s);
|
|
if (sess.events.find(code)!=sess.events.end()) {
|
|
vector<Event *>::reverse_iterator i=sess.events[code].rbegin();
|
|
assert(i!=sess.events[code].rend());
|
|
tmp=(*i)->time();
|
|
if (fir) {
|
|
date=tmp;
|
|
fir=false;
|
|
} else {
|
|
if (tmp>date) date=tmp;
|
|
}
|
|
}
|
|
}
|
|
return date;
|
|
}
|
|
|
|
void Day::OpenEvents()
|
|
{
|
|
vector<Session *>::iterator s;
|
|
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
(*s)->OpenEvents();
|
|
}
|
|
|
|
}
|
|
void Day::OpenWaveforms()
|
|
{
|
|
vector<Session *>::iterator s;
|
|
|
|
for (s=sessions.begin();s!=sessions.end();s++) {
|
|
(*s)->OpenWaveforms();
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// Event implmementation
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
Event::Event(wxDateTime time,MachineCode code, EventDataType * data, int fields)
|
|
:e_time(time),e_code(code)
|
|
{
|
|
e_fields=fields;
|
|
for (int i=0; i<fields; i++) {
|
|
e_data.push_back(data[i]);
|
|
}
|
|
|
|
}
|
|
Event::~Event()
|
|
{
|
|
|
|
};
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// Waveform implmementation
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
Waveform::Waveform(wxDateTime time,MachineCode code, SampleFormat *data,int samples,float duration,SampleFormat min, SampleFormat max)
|
|
:w_time(time),w_code(code),w_data(data),w_samples(samples),w_duration(duration)
|
|
{
|
|
w_totalspan=wxTimeSpan::Seconds(duration);
|
|
double rate=duration/samples;
|
|
w_samplespan=wxTimeSpan::Milliseconds(rate*1000.0);
|
|
Min=min;
|
|
Max=max;
|
|
}
|
|
Waveform::~Waveform()
|
|
{
|
|
delete [] w_data;
|
|
}
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// CPAP implmementation
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CPAP::CPAP(Profile *p,MachineID id):Machine(p,id)
|
|
{
|
|
m_type=MT_CPAP;
|
|
|
|
// FlagColours=DefaultFlagColours;
|
|
}
|
|
|
|
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()
|
|
{
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// Session implmementation
|
|
// This stuff contains the base calculation smarts
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
Session::Session(Machine * m,SessionID session)
|
|
{
|
|
s_machine=m;
|
|
s_session=session;
|
|
s_changed=false;
|
|
s_events_loaded=false;
|
|
s_waves_loaded=false;
|
|
s_wavefile=wxEmptyString;
|
|
s_eventfile=wxEmptyString;
|
|
}
|
|
Session::~Session()
|
|
{
|
|
TrashEvents();
|
|
TrashWaveforms();
|
|
}
|
|
double Session::min_event_field(MachineCode mc,int field)
|
|
{
|
|
if (events.find(mc)==events.end()) return 0;
|
|
|
|
bool first=true;
|
|
double min;
|
|
vector<Event *>::iterator i;
|
|
for (i=events[mc].begin(); i!=events[mc].end(); i++) {
|
|
if (field>(*i)->e_fields) throw BoundsError();
|
|
if (first) {
|
|
first=false;
|
|
min=(*(*i))[field];
|
|
} else {
|
|
if (min>(*(*i))[field]) min=(*(*i))[field];
|
|
}
|
|
}
|
|
return min;
|
|
}
|
|
double Session::max_event_field(MachineCode mc,int field)
|
|
{
|
|
if (events.find(mc)==events.end()) return 0;
|
|
|
|
bool first=true;
|
|
double max;
|
|
vector<Event *>::iterator i;
|
|
for (i=events[mc].begin(); i!=events[mc].end(); i++) {
|
|
if (field>(*i)->e_fields) throw BoundsError();
|
|
if (first) {
|
|
first=false;
|
|
max=(*(*i))[field];
|
|
} else {
|
|
if (max<(*(*i))[field]) max=(*(*i))[field];
|
|
}
|
|
}
|
|
return max;
|
|
}
|
|
|
|
double Session::sum_event_field(MachineCode mc,int field)
|
|
{
|
|
if (events.find(mc)==events.end()) return 0;
|
|
|
|
double sum=0;
|
|
vector<Event *>::iterator i;
|
|
|
|
for (i=events[mc].begin(); i!=events[mc].end(); i++) {
|
|
if (field>(*i)->e_fields) throw BoundsError();
|
|
sum+=(*(*i))[field];
|
|
}
|
|
return sum;
|
|
}
|
|
double Session::avg_event_field(MachineCode mc,int field)
|
|
{
|
|
if (events.find(mc)==events.end()) return 0;
|
|
|
|
double sum=0;
|
|
int cnt=0;
|
|
vector<Event *>::iterator i;
|
|
|
|
for (i=events[mc].begin(); i!=events[mc].end(); i++) {
|
|
if (field>(*i)->e_fields) throw BoundsError();
|
|
sum+=(*(*i))[field];
|
|
cnt++;
|
|
}
|
|
|
|
return sum/cnt;
|
|
}
|
|
|
|
bool sortfunction (double i,double j) { return (i<j); }
|
|
|
|
double Session::percentile(MachineCode mc,int field,double percent)
|
|
{
|
|
if (events.find(mc)==events.end()) return 0;
|
|
|
|
vector<EventDataType> array;
|
|
|
|
vector<Event *>::iterator e;
|
|
|
|
for (e=events[mc].begin(); e!=events[mc].end(); e++) {
|
|
Event & ev = *(*e);
|
|
array.push_back(ev[0]);
|
|
}
|
|
std::sort(array.begin(),array.end(),sortfunction);
|
|
int size=array.size();
|
|
double i=size*percent;
|
|
double t;
|
|
double q=modf(i,&t);
|
|
int j=t;
|
|
if (j>=size-1) return array[j];
|
|
|
|
double a=array[j-1];
|
|
double b=array[j];
|
|
if (a==b) return a;
|
|
double c=(b-a);
|
|
double d=c*q;
|
|
return array[j]+q;
|
|
}
|
|
|
|
double Session::weighted_avg_event_field(MachineCode mc,int field)
|
|
{
|
|
if (events.find(mc)==events.end()) return 0;
|
|
|
|
int cnt=0;
|
|
|
|
bool first=true;
|
|
wxDateTime last;
|
|
int lastval,val;
|
|
const int max_slots=2600;
|
|
wxTimeSpan vtime[max_slots]=wxTimeSpan(0);
|
|
|
|
|
|
double mult;
|
|
if ((mc==CPAP_Pressure) || (mc==CPAP_EAP) || (mc==CPAP_IAP)) {
|
|
mult=10.0;
|
|
|
|
} else mult=10.0;
|
|
|
|
vector<Event *>::iterator i;
|
|
|
|
for (i=events[mc].begin(); i!=events[mc].end(); i++) {
|
|
Event & e =(*(*i));
|
|
val=e[field]*mult;
|
|
if (field > e.e_fields) throw BoundsError();
|
|
if (first) {
|
|
first=false;
|
|
} else {
|
|
wxTimeSpan d=e.e_time-last;
|
|
if (lastval>max_slots) {
|
|
wxLogError(wxT("max_slots to small in Session::weighted_avg_event_fied()"));
|
|
}
|
|
|
|
vtime[lastval]+=d;
|
|
|
|
}
|
|
cnt++;
|
|
last=e.e_time;
|
|
lastval=val;
|
|
}
|
|
|
|
wxTimeSpan total(0);
|
|
for (int i=0; i<max_slots; i++) total+=vtime[i];
|
|
//double hours=total.GetSeconds().GetLo()/3600.0;
|
|
|
|
double s0=0,s1=0,s2=0;
|
|
|
|
for (int i=0; i<max_slots; i++) {
|
|
if (vtime[i]>wxTimeSpan(0)) {
|
|
s0=(vtime[i].GetSeconds().GetLo()/3600.0);
|
|
s1+=i*s0;
|
|
s2+=s0;
|
|
}
|
|
}
|
|
|
|
return (s1/hours())/mult;
|
|
}
|
|
|
|
void Session::AddEvent(Event * e)
|
|
{
|
|
events[e->code()].push_back(e);
|
|
if (e->time()>s_last) s_last=e->time();
|
|
// wxLogMessage(e->time().Format(wxT("%Y-%m-%d %H:%M:%S"))+wxString::Format(wxT(" %04i %02i "),e->code(),e->fields()));
|
|
}
|
|
void Session::AddWaveform(Waveform *w)
|
|
{
|
|
waveforms[w->code()].push_back(w);
|
|
//wxLogMessage(w->start().Format(wxT("%Y-%m-%d %H:%M:%S"))+wxT(" ")+w->end().Format(wxT("%Y-%m-%d %H:%M:%S"))+wxString::Format(wxT(" %i %.1f"),w->samples(), w->duration()));
|
|
|
|
// Could parse the flow data for Breaths per minute info here..
|
|
|
|
|
|
}
|
|
void Session::TrashEvents()
|
|
// Trash this sessions Events and release memory.
|
|
{
|
|
map<MachineCode,vector<Event *> >::iterator i;
|
|
vector<Event *>:: iterator j;
|
|
for (i=events.begin(); i!=events.end(); i++) {
|
|
for (j=i->second.begin(); j!=i->second.end(); j++) {
|
|
delete *j;
|
|
}
|
|
}
|
|
events.clear();
|
|
}
|
|
void Session::TrashWaveforms()
|
|
// Trash this sessions Waveforms and release memory.
|
|
{
|
|
map<MachineCode,vector<Waveform *> >::iterator i;
|
|
vector<Waveform *>:: iterator j;
|
|
for (i=waveforms.begin(); i!=waveforms.end(); i++) {
|
|
for (j=i->second.begin(); j!=i->second.end(); j++) {
|
|
delete *j;
|
|
}
|
|
}
|
|
waveforms.clear();
|
|
}
|
|
|
|
|
|
const int max_pack_size=128;
|
|
template <class T>
|
|
int pack(char * buf,T * var)
|
|
{
|
|
int c=0;
|
|
int s=sizeof(T);
|
|
if (s>max_pack_size) return 0;
|
|
|
|
char * dat=(char *)var;
|
|
if (wxIsPlatformLittleEndian()) {
|
|
for (int i=0; i<s; i++) {
|
|
buf[i]=dat[i];
|
|
}
|
|
} else {
|
|
for (int i=0; i<s; i++) {
|
|
buf[s-i]=dat[i];
|
|
}
|
|
}
|
|
return s;
|
|
}
|
|
|
|
/*bool Session::Load(wxString path) // Not Used..
|
|
// Load Session Data from our format
|
|
// {DataDir}/{MachineID}/{SessionID}.{ext}
|
|
{
|
|
path=path+wxFileName::GetPathSeparator()+s_machine->hexid();
|
|
wxString base;
|
|
base.Printf(wxT("%08x"),s_session);
|
|
base=path+wxFileName::GetPathSeparator()+base;
|
|
wxPuts(wxT("Loading Session: ")+base);
|
|
bool a,b,c;
|
|
summary.clear();
|
|
TrashEvents();
|
|
TrashWaveforms();
|
|
//try...
|
|
a=LoadSummary(base+wxT(".000"));
|
|
b=LoadEvents(base+wxT(".001"));
|
|
c=LoadWaveforms(base+wxT(".002"));
|
|
return a;
|
|
}*/
|
|
|
|
bool Session::Store(wxString path)
|
|
// Storing Session Data in our format
|
|
// {DataDir}/{MachineID}/{SessionID}.{ext}
|
|
{
|
|
|
|
// path=path+wxFileName::GetPathSeparator()+s_machine->hexid();
|
|
if (!wxDirExists(path)) wxMkdir(path);
|
|
wxString base;
|
|
base.Printf(wxT("%08lx"),s_session);
|
|
base=path+wxFileName::GetPathSeparator()+base;
|
|
//wxPuts(wxT("Storing Session: ")+base);
|
|
bool a,b,c;
|
|
a=StoreSummary(base+wxT(".000"));
|
|
b=StoreEvents(base+wxT(".001"));
|
|
c=StoreWaveforms(base+wxT(".002"));
|
|
return a&b&c;
|
|
}
|
|
|
|
const wxUint32 magic=0xC73216AB;
|
|
|
|
bool Session::StoreSummary(wxString filename)
|
|
{
|
|
BinaryFile f;
|
|
f.Open(filename,BF_WRITE);
|
|
|
|
f.Pack((wxUint32)magic); // Magic Number
|
|
f.Pack((wxUint32)s_machine->id()); // Machine ID
|
|
f.Pack((wxUint32)s_session); // Session ID
|
|
f.Pack((wxUint16)0); // File Type 0 == Summary File
|
|
f.Pack((wxUint16)0); // File Version
|
|
|
|
time_t starttime=s_first.GetTicks();
|
|
time_t duration=s_last.GetTicks()-starttime;
|
|
|
|
f.Pack(s_first); // Session Start Time
|
|
f.Pack((wxInt16)duration); // Duration of sesion in seconds.
|
|
f.Pack((wxInt16)summary.size());
|
|
|
|
map<MachineCode,MCDataType> mctype;
|
|
|
|
// First output the Machine Code and type for each summary record
|
|
map<MachineCode,wxVariant>::iterator i;
|
|
for (i=summary.begin(); i!=summary.end(); i++) {
|
|
MachineCode mc=i->first;
|
|
|
|
wxString type=i->second.GetType(); // Urkk.. this is a mess.
|
|
if (type==wxT("bool")) {
|
|
mctype[mc]=MC_bool;
|
|
} else if (type==wxT("long")) {
|
|
mctype[mc]=MC_long;
|
|
} else if (type==wxT("float")) {
|
|
mctype[mc]=MC_float;
|
|
} else if (type==wxT("double")) {
|
|
mctype[mc]=MC_double;
|
|
} else if (type==wxT("string")) {
|
|
mctype[mc]=MC_string;
|
|
} else if (type==wxT("datetime")) {
|
|
mctype[mc]=MC_string;
|
|
} else {
|
|
wxPuts(wxT("Error in Session->StoreSummary: Can't pack variant type ")+type);
|
|
exit(1);
|
|
}
|
|
f.Pack((wxInt16)mc);
|
|
f.Pack((wxInt8)mctype[mc]);
|
|
}
|
|
// Then dump out the actual data, according to format.
|
|
for (i=summary.begin(); i!=summary.end(); i++) {
|
|
MachineCode mc=i->first;
|
|
if (mctype[mc]==MC_bool) {
|
|
f.Pack((wxInt8)i->second.GetBool());
|
|
} else if (mctype[mc]==MC_long) {
|
|
f.Pack((wxInt32)i->second.GetLong());
|
|
} else if (mctype[mc]==MC_float) {
|
|
f.Pack((float)i->second.GetDouble()); // gah
|
|
} else if (mctype[mc]==MC_double) {
|
|
f.Pack((double)i->second.GetDouble());
|
|
} else if (mctype[mc]==MC_string) {
|
|
f.Pack(i->second.GetString());
|
|
} else if (mctype[mc]==MC_datetime) {
|
|
f.Pack(i->second.GetDateTime());
|
|
}
|
|
}
|
|
f.Close();
|
|
wxFFile j;
|
|
return true;
|
|
}
|
|
bool Session::LoadSummary(wxString filename)
|
|
{
|
|
//wxPuts(wxT("Loading Summary ")+filename);
|
|
BinaryFile f;
|
|
if (!f.Open(filename,BF_READ)) {
|
|
wxPuts(wxT("Couldn't open file")+filename);
|
|
return false;
|
|
}
|
|
|
|
wxUint32 t32;
|
|
wxUint16 t16;
|
|
wxUint8 t8;
|
|
|
|
wxInt16 sumsize;
|
|
map<MachineCode,MCDataType> mctype;
|
|
vector<MachineCode> mcorder;
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // Magic Number
|
|
if (t32!=magic) throw UnpackError();
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // MachineID
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // Sessionid;
|
|
s_session=t32;
|
|
|
|
if (!f.Unpack(t16)) throw UnpackError(); // File Type
|
|
if (t16!=0) throw UnpackError(); //wrong file type
|
|
|
|
if (!f.Unpack(t16)) throw UnpackError(); // File Version
|
|
// dont care yet
|
|
|
|
if (!f.Unpack(s_first)) throw UnpackError(); // Start time
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Duration // (16bit==Limited to 18 hours)
|
|
|
|
s_last=s_first+wxTimeSpan::Seconds(t16);
|
|
s_hours=t16/3600.0;
|
|
|
|
if (!f.Unpack(sumsize)) throw UnpackError(); // Summary size (number of Machine Code lists)
|
|
|
|
for (int i=0; i<sumsize; i++) {
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Machine Code
|
|
MachineCode mc=(MachineCode)t16;
|
|
if (!f.Unpack(t8)) throw UnpackError(); // Data Type
|
|
mctype[mc]=(MCDataType)t8;
|
|
mcorder.push_back(mc);
|
|
}
|
|
|
|
for (int i=0; i<sumsize; i++) { // Load each summary entry according to type
|
|
MachineCode mc=mcorder[i];
|
|
if (mctype[mc]==MC_bool) {
|
|
bool b;
|
|
if (!f.Unpack(b)) throw UnpackError();
|
|
summary[mc]=b;
|
|
} else if (mctype[mc]==MC_long) {
|
|
if (!f.Unpack(t32)) throw UnpackError();
|
|
summary[mc]=(long)t32;
|
|
} else if (mctype[mc]==MC_float) {
|
|
float fl;
|
|
if (!f.Unpack(fl)) throw UnpackError();
|
|
summary[mc]=(double)fl;
|
|
} else if (mctype[mc]==MC_double) {
|
|
double dl;
|
|
if (!f.Unpack(dl)) throw UnpackError();
|
|
summary[mc]=(double)dl;
|
|
} else if (mctype[mc]==MC_string) {
|
|
wxString s;
|
|
if (!f.Unpack(s)) throw UnpackError();
|
|
summary[mc]=s;
|
|
} else if (mctype[mc]==MC_datetime) {
|
|
wxDateTime dt;
|
|
if (!f.Unpack(dt)) throw UnpackError();
|
|
summary[mc]=dt;
|
|
}
|
|
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Session::StoreEvents(wxString filename)
|
|
{
|
|
BinaryFile f;
|
|
f.Open(filename,BF_WRITE);
|
|
|
|
f.Pack((wxUint32)magic); // Magic Number
|
|
f.Pack((wxUint32)s_machine->id()); // Machine ID
|
|
f.Pack((wxUint32)s_session); // This session's ID
|
|
f.Pack((wxUint16)1); // File type 1 == Event
|
|
f.Pack((wxUint16)0); // File Version
|
|
|
|
time_t starttime=s_first.GetTicks();
|
|
time_t duration=s_last.GetTicks()-starttime;
|
|
|
|
f.Pack(s_first);
|
|
f.Pack((wxInt16)duration);
|
|
|
|
f.Pack((wxInt16)events.size()); // Number of event categories
|
|
|
|
|
|
map<MachineCode,vector<Event *> >::iterator i;
|
|
vector<Event *>::iterator j;
|
|
|
|
for (i=events.begin(); i!=events.end(); i++) {
|
|
f.Pack((wxInt16)i->first); // MachineID
|
|
f.Pack((wxInt16)i->second.size()); // count of events in this category
|
|
j=i->second.begin();
|
|
f.Pack((wxInt8)(*j)->fields()); // number of data fields in this event type
|
|
}
|
|
bool first;
|
|
float tf;
|
|
time_t last,eventtime,delta;
|
|
|
|
for (i=events.begin(); i!=events.end(); i++) {
|
|
first=true;
|
|
for (j=i->second.begin(); j!=i->second.end(); j++) {
|
|
eventtime=(*j)->time().GetTicks();
|
|
if (first) {
|
|
f.Pack((*j)->time());
|
|
first=false;
|
|
} else {
|
|
delta=eventtime-last;
|
|
if (delta>0xffff) {
|
|
wxPuts(wxT("StoreEvent: Delta too big.. needed to use bigger value"));
|
|
exit(1);
|
|
}
|
|
f.Pack((wxInt16)delta);
|
|
}
|
|
for (int k=0; k<(*j)->fields(); k++) {
|
|
tf=(*(*j))[k];
|
|
f.Pack((float)tf);
|
|
}
|
|
last=eventtime;
|
|
}
|
|
}
|
|
f.Close();
|
|
return true;
|
|
}
|
|
bool Session::LoadEvents(wxString filename)
|
|
{
|
|
BinaryFile f;
|
|
if (!f.Open(filename,BF_READ)) {
|
|
wxPuts(wxT("Couldn't open file")+filename);
|
|
return false;
|
|
}
|
|
|
|
wxUint32 t32;
|
|
wxUint16 t16;
|
|
wxUint8 t8;
|
|
wxInt16 i16;
|
|
|
|
// wxInt16 sumsize;
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // Magic Number
|
|
if (t32!=magic) throw UnpackError();
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // MachineID
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // Sessionid;
|
|
s_session=t32;
|
|
|
|
if (!f.Unpack(t16)) throw UnpackError(); // File Type
|
|
if (t16!=1) throw UnpackError(); //wrong file type
|
|
|
|
if (!f.Unpack(t16)) throw UnpackError(); // File Version
|
|
// dont give a crap yet..
|
|
|
|
if (!f.Unpack(s_first)) throw UnpackError(); // Start time
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Duration // (16bit==Limited to 18 hours)
|
|
|
|
s_last=s_first+wxTimeSpan::Seconds(t16);
|
|
s_hours=t16/3600.0;
|
|
|
|
wxInt16 evsize;
|
|
if (!f.Unpack(evsize)) throw UnpackError(); // Summary size (number of Machine Code lists)
|
|
|
|
map<MachineCode,wxInt16> mcsize;
|
|
map<MachineCode,wxInt16> mcfields;
|
|
vector<MachineCode> mcorder;
|
|
|
|
MachineCode mc;
|
|
for (int i=0; i<evsize; i++) {
|
|
if (!f.Unpack(t16)) throw UnpackError(); // MachineID
|
|
mc=(MachineCode)t16;
|
|
mcorder.push_back(mc);
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Count of this Event
|
|
mcsize[mc]=t16;
|
|
if (!f.Unpack(t8)) throw UnpackError(); // Count of this Event
|
|
mcfields[mc]=t8;
|
|
}
|
|
|
|
for (int i=0; i<evsize; i++) {
|
|
mc=mcorder[i];
|
|
bool first=true;
|
|
wxDateTime d;
|
|
float fl;
|
|
events[mc].clear();
|
|
for (int e=0; e<mcsize[mc]; e++) {
|
|
if (first) {
|
|
if (!f.Unpack(d)) throw UnpackError(); // Timestamp
|
|
first=false;
|
|
} else {
|
|
if (!f.Unpack(i16)) throw UnpackError(); // Time Delta
|
|
d+=wxTimeSpan::Seconds(i16);
|
|
}
|
|
EventDataType ED[max_number_event_fields];
|
|
for (int c=0; c<mcfields[mc]; c++) {
|
|
if (!f.Unpack(fl)); //throw UnpackError(); // Data Fields in float format
|
|
ED[c]=fl;
|
|
}
|
|
Event *ev=new Event(d,mc,ED,mcfields[mc]);
|
|
|
|
AddEvent(ev);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
bool Session::StoreWaveforms(wxString filename)
|
|
{
|
|
BinaryFile f;
|
|
f.Open(filename,BF_WRITE);
|
|
|
|
wxUint16 t16;
|
|
|
|
f.Pack((wxUint32)magic); // Magic Number
|
|
f.Pack((wxUint32)s_machine->id()); // Machine ID
|
|
f.Pack((wxUint32)s_session); // This session's ID
|
|
f.Pack((wxUint16)2); // File type 2 == Waveform
|
|
f.Pack((wxUint16)0); // File Version
|
|
|
|
time_t starttime=s_first.GetTicks();
|
|
time_t duration=s_last.GetTicks()-starttime;
|
|
|
|
f.Pack(s_first);
|
|
f.Pack((wxInt16)duration);
|
|
|
|
f.Pack((wxInt16)waveforms.size()); // Number of different waveforms
|
|
|
|
map<MachineCode,vector<Waveform *> >::iterator i;
|
|
vector<Waveform *>::iterator j;
|
|
for (i=waveforms.begin(); i!=waveforms.end(); i++) {
|
|
f.Pack((wxInt16)i->first); // Machine Code
|
|
t16=i->second.size();
|
|
f.Pack(t16); // Number of (hopefully non-linear) waveform chunks
|
|
|
|
for (j=i->second.begin(); j!=i->second.end(); j++) {
|
|
|
|
Waveform &w=*(*j);
|
|
f.Pack(w.start()); // Start time of first waveform chunk
|
|
|
|
//wxInt32 samples;
|
|
//double seconds;
|
|
|
|
f.Pack((wxInt32)w.samples()); // Total number of samples
|
|
f.Pack((float)w.duration()); // Total number of seconds
|
|
f.Pack((wxInt16)w.min());
|
|
f.Pack((wxInt16)w.max());
|
|
f.Pack((wxInt8)sizeof(SampleFormat)); // Bytes per sample
|
|
f.Pack((wxInt8)0); // signed.. all samples for now are signed 16bit.
|
|
//t8=0; // 0=signed, 1=unsigned, 2=float
|
|
|
|
// followed by sample data.
|
|
if (wxIsPlatformLittleEndian()) {
|
|
f.Write(w.GetBuffer(),w.samples()*sizeof(SampleFormat));
|
|
} else {
|
|
for (int k=0; k<(*j)->samples(); k++) f.Pack((wxInt16)w[k]);
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
bool Session::LoadWaveforms(wxString filename)
|
|
{
|
|
BinaryFile f;
|
|
if (!f.Open(filename,BF_READ)) {
|
|
wxPuts(wxT("Couldn't open file")+filename);
|
|
return false;
|
|
}
|
|
|
|
wxUint32 t32;
|
|
wxUint16 t16;
|
|
wxUint8 t8;
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // Magic Number
|
|
if (t32!=magic) throw UnpackError();
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // MachineID
|
|
|
|
if (!f.Unpack(t32)) throw UnpackError(); // Sessionid;
|
|
s_session=t32;
|
|
|
|
if (!f.Unpack(t16)) throw UnpackError(); // File Type
|
|
if (t16!=2) throw UnpackError(); //wrong file type?
|
|
|
|
if (!f.Unpack(t16)) throw UnpackError(); // File Version
|
|
// dont give a crap yet..
|
|
|
|
if (!f.Unpack(s_first)) throw UnpackError(); // Start time
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Duration // (16bit==Limited to 18 hours)
|
|
|
|
s_last=s_first+wxTimeSpan::Seconds(t16);
|
|
s_hours=t16/3600.0;
|
|
|
|
wxInt16 wvsize;
|
|
if (!f.Unpack(wvsize)) throw UnpackError(); // Summary size (number of Machine Code lists)
|
|
|
|
MachineCode mc;
|
|
wxDateTime date;
|
|
float seconds;
|
|
wxInt32 samples;
|
|
int chunks;
|
|
SampleFormat min,max;
|
|
for (int i=0; i<wvsize; i++) {
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Machine Code
|
|
mc=(MachineCode)t16;
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Machine Code
|
|
chunks=t16;
|
|
|
|
for (int i=0; i<chunks; i++) {
|
|
|
|
if (!f.Unpack(date)) throw UnpackError(); // Waveform DateTime
|
|
if (!f.Unpack(samples)) throw UnpackError(); // Number of samples
|
|
if (!f.Unpack(seconds)) throw UnpackError(); // Duration in seconds
|
|
if (!f.Unpack(min)) throw UnpackError(); // Min value
|
|
if (!f.Unpack(max)) throw UnpackError(); // Min value
|
|
if (!f.Unpack(t8)) throw UnpackError(); // sample size in bytes
|
|
if (!f.Unpack(t8)) throw UnpackError(); // Number of samples
|
|
|
|
SampleFormat *data=new SampleFormat [samples];
|
|
|
|
for (int k=0; k<samples; k++) {
|
|
if (!f.Unpack(t16)) throw UnpackError(); // Number of samples
|
|
data[k]=t16;
|
|
}
|
|
Waveform *w=new Waveform(date,mc,data,samples,seconds,min,max);
|
|
AddWaveform(w);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
|