OSCAR-code/oximetry.cpp

625 lines
18 KiB
C++
Raw Normal View History

2011-07-13 05:27:27 +00:00
#include <QDebug>
#include <QProgressBar>
2011-07-20 03:43:41 +00:00
#include <QMessageBox>
2011-07-27 09:21:53 +00:00
/*#include "oximetry.h"
2011-07-13 05:27:27 +00:00
#include "ui_oximetry.h"
#include "qextserialport/qextserialenumerator.h"
#include "SleepLib/loader_plugins/cms50_loader.h"
#include "Graphs/gXAxis.h"
#include "Graphs/gBarChart.h"
#include "Graphs/gLineChart.h"
#include "Graphs/gYAxis.h"
#include "Graphs/gFooBar.h"
2011-07-13 05:27:27 +00:00
Oximetry::Oximetry(QWidget *parent,QGLWidget * shared) :
2011-07-13 05:27:27 +00:00
QWidget(parent),
ui(new Ui::Oximetry)
{
ui->setupUi(this);
port=NULL;
2011-07-18 02:33:25 +00:00
portname="";
QString prof=pref["Profile"].toString();
profile=Profiles::Get(prof);
if (!profile) {
qWarning("Couldn't get profile.. Have to abort!");
abort();
}
gSplitter=new QSplitter(Qt::Vertical,ui->scrollArea);
gSplitter->setStyleSheet("QSplitter::handle { background-color: 'dark grey'; }");
gSplitter->setHandleWidth(2);
ui->graphLayout->addWidget(gSplitter);
2011-07-18 02:33:25 +00:00
AddData(plethy=new WaveData(OXI_Plethy));
AddGraph(PLETHY=new gGraphWindow(gSplitter,tr("Plethysomogram"),shared));
2011-07-18 02:33:25 +00:00
AddData(pulse=new EventData(OXI_Pulse));
2011-07-18 02:33:25 +00:00
AddGraph(PULSE=new gGraphWindow(gSplitter,tr("Pulse Rate"),PLETHY));
AddData(spo2=new EventData(OXI_SPO2));
2011-07-18 02:33:25 +00:00
AddGraph(SPO2=new gGraphWindow(gSplitter,tr("SPO2"),PLETHY));
2011-07-18 02:33:25 +00:00
plethy->SetRealMaxY(128);
pulse->SetRealMaxY(130);
spo2->SetRealMaxY(100);
2011-07-18 02:33:25 +00:00
for (unsigned i=0;i<Data.size();i++) {
Data[i]->AddSegment(1000000);
Data[i]->np.push_back(0);
Data[i]->SetRealMinY(0);
Data[i]->SetMinY(0);
Data[i]->SetMaxY(Data[i]->RealMaxY());
}
2011-07-18 02:33:25 +00:00
for (unsigned i=0;i<Graphs.size();i++) {
for (unsigned j=0;j<Graphs.size();j++) {
if (Graphs[i]!=Graphs[j])
Graphs[i]->LinkZoom(Graphs[j]);
}
2011-07-18 02:33:25 +00:00
Graphs[i]->AddLayer(new gYAxis());
Graphs[i]->AddLayer(new gXAxis());
Graphs[i]->AddLayer(new gFooBar());
2011-07-18 10:57:34 +00:00
gSplitter->addWidget(Graphs[i]);
}
SPO2->AddLayer(new gLineChart(spo2,Qt::blue,65536,false,false,false));
PULSE->AddLayer(new gLineChart(pulse,Qt::red,65536,false,false,false));
PLETHY->AddLayer(new gLineChart(plethy,Qt::black,65536,true,false,false));
for (unsigned i=0;i<Graphs.size();i++) {
2011-07-18 02:33:25 +00:00
Graphs[i]->setMinimumHeight(150);
Graphs[i]->SetSplitter(gSplitter);
Graphs[i]->RealMinY();
Graphs[i]->RealMaxY();
Graphs[i]->MinY();
Graphs[i]->MaxY();
}
2011-07-13 05:27:27 +00:00
on_RefreshPortsButton_clicked();
}
Oximetry::~Oximetry()
{
delete ui;
}
void Oximetry::on_RefreshPortsButton_clicked()
{
QList<QextPortInfo> ports = QextSerialEnumerator::getPorts();
ui->SerialPortsCombo->clear();
int z=0;
QString firstport;
bool current_found=false;
2011-07-19 16:44:03 +00:00
#ifdef Q_WS_WIN32
2011-07-18 11:56:56 +00:00
#define qesPORTNAME portName
#else
#define qesPORTNAME physName
#endif
2011-07-13 05:27:27 +00:00
for (int i = 0; i < ports.size(); i++) {
if (ports.at(i).friendName.toUpper().contains("USB")) {
2011-07-18 11:56:56 +00:00
if (firstport.isEmpty()) firstport=ports.at(i). qesPORTNAME;
if (!portname.isEmpty() && ports.at(i).qesPORTNAME==portname) current_found=true;
ui->SerialPortsCombo->addItem(ports.at(i).qesPORTNAME);
z++;
qDebug() << "Serial Port:" << ports.at(i).qesPORTNAME << ports.at(i).friendName;
}
2011-07-13 05:27:27 +00:00
//qDebug() << "port name:" << ports.at(i).portName;
//qDebug() << "enumerator name:" << ports.at(i).enumName;
}
if (z>0) {
ui->RunButton->setEnabled(true);
ui->ImportButton->setEnabled(true);
if (!current_found) portname=firstport;
} else {
ui->RunButton->setEnabled(false);
ui->ImportButton->setEnabled(false);
portname="";
}
}
void Oximetry::RedrawGraphs()
{
2011-07-18 02:33:25 +00:00
for (vector<gGraphWindow *>::iterator g=Graphs.begin();g!=Graphs.end();g++) {
(*g)->updateGL();
}
}
void Oximetry::on_RunButton_toggled(bool checked)
{
if (checked) {
lasttime=0;
lastpulse=-1;
lastspo2=-1;
plethy->np[0]=0;
2011-07-17 12:10:51 +00:00
lasttime=QDateTime::currentMSecsSinceEpoch();
starttime=lasttime;
plethy->SetRealMinX(double(lasttime)/86400000.0);
plethy->SetRealMaxX(double(lasttime+60000)/86400000.0);
plethy->SetMinX(double(lasttime)/86400000.0);
plethy->SetMaxX(double(lasttime+30000)/86400000.0);
PLETHY->RealMinX();
PLETHY->RealMaxX();
2011-07-18 02:33:25 +00:00
PLETHY->MinX();
PLETHY->MaxX();
2011-07-17 12:10:51 +00:00
plethy->SetReady(true);
plethy->SetVC(1);
plethy->np[0]=0;
pulse->SetRealMinX(double(lasttime)/86400000.0);
pulse->SetRealMaxX(double(lasttime)/86400000.0+(1.0/24.0));
pulse->SetMinX(double(lasttime)/86400000.0);
pulse->SetMaxX(double(lasttime)/86400000.0+(1.0/24.0));
pulse->np[0]=0;
pulse->SetReady(true);
pulse->SetVC(1);
PULSE->RealMinX();
PULSE->RealMaxX();
2011-07-18 02:33:25 +00:00
PULSE->MinX();
PULSE->MaxX();
2011-07-17 12:10:51 +00:00
spo2->SetRealMinX(double(lasttime)/86400000.0);
spo2->SetRealMaxX(double(lasttime)/86400000.0+(1.0/24.0));
spo2->SetMinX(double(lasttime)/86400000.0);
spo2->SetMaxX(double(lasttime)/86400000.0+(1.0/24.0));
spo2->np[0]=0;
spo2->SetReady(true);
spo2->SetVC(1);
SPO2->RealMinX();
SPO2->RealMaxX();
2011-07-18 02:33:25 +00:00
SPO2->MinX();
SPO2->MaxX();
ui->RunButton->setText("&Stop");
ui->SerialPortsCombo->setEnabled(false);
// Disconnect??
port=new QextSerialPort(portname,QextSerialPort::EventDriven);
port->setBaudRate(BAUD19200);
port->setFlowControl(FLOW_OFF);
port->setParity(PAR_ODD);
port->setDataBits(DATA_8);
port->setStopBits(STOP_1);
if (port->open(QIODevice::ReadWrite) == true) {
connect(port, SIGNAL(readyRead()), this, SLOT(onReadyRead()));
connect(port, SIGNAL(dsrChanged(bool)), this, SLOT(onDsrChanged(bool)));
if (!(port->lineStatus() & LS_DSR))
qDebug() << "warning: device is not turned on";
qDebug() << "listening for data on" << port->portName();
} else {
qDebug() << "device failed to open:" << port->errorString();
}
portmode=PM_LIVE;
} else {
ui->RunButton->setText("&Start");
ui->SerialPortsCombo->setEnabled(true);
delete port;
port=NULL;
spo2->point[0][spo2->np[0]].setX(lasttime/86400000.0);
spo2->point[0][spo2->np[0]++].setY(lastspo2);
pulse->point[0][pulse->np[0]].setX(lasttime/86400000.0);
pulse->point[0][pulse->np[0]++].setY(lastpulse);
pulse->SetRealMaxX(plethy->RealMaxX());
spo2->SetRealMaxX(plethy->RealMaxX());
spo2->SetMinX(plethy->RealMinX());
spo2->SetMaxX(plethy->RealMaxX());
pulse->SetMinX(plethy->RealMinX());
pulse->SetMaxX(plethy->RealMaxX());
2011-07-17 10:00:26 +00:00
plethy->SetMinX(plethy->RealMinX());
PULSE->RealMaxX();
PULSE->MaxX();
PULSE->MinX();
SPO2->RealMaxX();
SPO2->MaxX();
SPO2->MinX();
2011-07-17 10:00:26 +00:00
PLETHY->MinX();
PLETHY->MaxX();
PLETHY->updateGL();
SPO2->updateGL();
PULSE->updateGL();
}
}
void Oximetry::on_SerialPortsCombo_activated(const QString &arg1)
{
portname=arg1;
}
void Oximetry::UpdatePlethy(qint8 d)
{
plethy->point[0][plethy->np[0]].setX(double(lasttime)/86400000.0);
plethy->point[0][plethy->np[0]++].setY(d);
lasttime+=20; // 50 samples per second?
plethy->SetRealMaxX(lasttime/86400000.0);
if (plethy->RealMaxX()-plethy->RealMinX()>(1.0/(24.0*120.0))) {
plethy->SetMinX(lasttime/86400000.0-(1.0/(24.0*120.0)));
plethy->SetMaxX(lasttime/86400000.0);
}
PLETHY->MinX();
PLETHY->MaxX();
PLETHY->RealMaxX();
pulse->SetRealMaxX(PLETHY->RealMaxX());
spo2->SetRealMaxX(PLETHY->RealMaxX());
PULSE->SetMinX(PLETHY->MinX());
SPO2->SetMinX(PLETHY->MinX());
PULSE->SetMaxX(PLETHY->MaxX());
SPO2->SetMaxX(PLETHY->MaxX());
if (plethy->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
plethy->np[0]=0;
lasttime=0;
}
//PLETHY->updateGL(); // Move this to a timer.
}
bool Oximetry::UpdatePulse(qint8 pul)
{
bool ret=false;
// Don't block zeros.. If the data is used, it's needed
// Can make the graph can skip them.
if (lastpulse!=pul)
{
pulse->point[0][pulse->np[0]].setX(double(lasttime)/86400000.0);
pulse->point[0][pulse->np[0]++].setY(pul);
if (pul!=0) {
if (pulse->MinY()==0) {
pulse->SetMinY((pul/10) * 10);
pulse->SetMaxY((pul/10+1) * 10);
PULSE->MinY();
PULSE->MaxY();
} else {
if (pul<pulse->MinY()) {
pulse->SetMinY((pul/10) * 10);
PULSE->MinY();
}
if (pul>pulse->MaxY()) {
pulse->SetMaxY((pul/10+1) * 10);
PULSE->MaxY();
}
}
}
// PULSE->updateGL();
if (pulse->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
pulse->np[0]=0;
lasttime=0;
}
ret=true;
//qDebug() << "Pulse=" << int(bytes[0]);
}
lastpulse=pul;
return ret;
}
bool Oximetry::UpdateSPO2(qint8 sp)
{
bool ret=false;
if (lastspo2!=sp)
{
spo2->point[0][spo2->np[0]].setX(double(lasttime)/86400000.0);
spo2->point[0][spo2->np[0]++].setY(sp);
if (sp!=0) {
if (spo2->MinY()==0) {
spo2->SetMinY((sp/10) * 10);
spo2->SetMaxY((sp/10+1) * 10);
SPO2->MinY();
SPO2->MaxY();
} else {
if (sp<spo2->MinY()) {
spo2->SetMinY((sp/10) * 10);
SPO2->MinY();
}
if (sp>spo2->MaxY()) {
spo2->SetMaxY((sp/10+1) * 10);
SPO2->MaxY();
}
}
}
//SPO2->updateGL();
if (spo2->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
spo2->np[0]=0;
lasttime=0;
}
ret=true;
//qDebug() << "SpO2=" << int(bytes[1]);
}
lastspo2=sp;
return ret;
}
void Oximetry::onReadyRead()
{
QByteArray bytes;
int a = port->bytesAvailable();
bytes.resize(a);
port->read(bytes.data(), bytes.size());
int i=0;
bool redraw_pulse,redraw_spo2;
redraw_pulse=redraw_spo2=false;
while (i<bytes.size()) {
if (bytes[i]&0x80) {
EventDataType d=bytes[i+1] & 0x7f;
UpdatePlethy(d);
i+=3;
} else {
if (UpdatePulse(bytes[i])) redraw_pulse=true;
if (UpdateSPO2(bytes[i+1])) redraw_spo2=true;
i+=2;
}
}
PLETHY->updateGL();
if (redraw_pulse)
PULSE->updateGL();
if (redraw_spo2)
SPO2->updateGL();
2011-07-27 09:21:53 +00:00
//if (bytes.size()==3) {
//} else if (bytes.size()==2) { // Data bytes in live mode
// Plethy data
2011-07-27 09:21:53 +00:00
//} else {
//qDebug() << "Got " << bytes.size() << " bytes";
2011-07-27 09:21:53 +00:00
//}
//QString aa=QString::number(bytes.size(),16)+"bytes: ";
//for (int i=0;i<bytes.size();i++) {
// aa+=" "+QString::number((unsigned char)bytes[i],16);
//}
//qDebug() << aa;
//lastsize=bytes.size();
}
void Oximetry::onDsrChanged(bool status)
{
if (status)
qDebug() << "device was turned on";
else
qDebug() << "device was turned off";
}
extern QProgressBar *qprogress;
extern QLabel *qstatus;
2011-07-18 10:57:34 +00:00
void DumpBytes(int blocks, unsigned char * b,int len)
{
2011-07-18 10:57:34 +00:00
QString a=QString::number(blocks,16)+": Bytes "+QString::number(len,16)+": ";
for (int i=0;i<len;i++) {
a.append(QString::number(b[i],16)+" ");
}
qDebug() << a;
}
// Move this code to CMS50 Importer??
void Oximetry::on_ImportButton_clicked()
{
2011-07-20 03:43:41 +00:00
QMessageBox msgbox(QMessageBox::Information,"Importing","Please Wait",QMessageBox::NoButton,this);
msgbox.show();
Machine *mach=profile->GetMachine(MT_OXIMETER);
if (!mach) {
CMS50Loader *l=dynamic_cast<CMS50Loader *>(GetLoader("CMS50"));
if (l) {
mach=l->CreateMachine(profile);
}
qDebug() << "Needed to create Oximeter device";
}
2011-07-20 03:43:41 +00:00
const int rb_size=0x200;
static unsigned char b1[2]={0xf5,0xf5};
static unsigned char b2[3]={0xf6,0xf6,0xf6};
static unsigned char rb[rb_size];
//b1[0]=0xf5;
//b1[1]=0xf5;
//b2[0]=0xf6;
//b2[1]=0xf6;
//b2[2]=0xf6;
unsigned char * buffer=NULL;
ui->SerialPortsCombo->setEnabled(false);
ui->RunButton->setText("&Start");
ui->RunButton->setChecked(false);
// port->write((char *)b1,2);
// return;
if (port) {
port->close();
delete port;
}
// Disconnect??
//qDebug() << "Initiating Polling Mode";
port=new QextSerialPort(portname,QextSerialPort::Polling);
port->setBaudRate(BAUD19200);
port->setFlowControl(FLOW_OFF);
port->setParity(PAR_ODD);
port->setDataBits(DATA_8);
port->setStopBits(STOP_1);
port->setTimeout(500);
if (port->open(QIODevice::ReadWrite) == true) {
// if (!(port->lineStatus() & LS_DSR))
// qDebug() << "warning: device is not turned on"; // CMS50 doesn't do this..
} else {
delete port;
port=NULL;
ui->SerialPortsCombo->setEnabled(true);
return;
}
bool done=false;
int res;
int blocks=0;
unsigned int bytes=0;
QString aa;
port->flush();
bool oneoff=false;
//qprogress->reset();
qstatus->setText("Importing");
qprogress->setValue(0);
qprogress->show();
int fails=0;
while (!done) {
if (port->write((char *)b1,2)==-1) {
qDebug() << "Couldn't write 2 lousy bytes to CMS50";
}
blocks=0;
int startpos=0;
unsigned int length=0;
2011-07-20 03:43:41 +00:00
int dr;
int ec;
do {
bool fnd=false;
2011-07-20 03:43:41 +00:00
dr=0;
ec=0;
do {
res=port->read((char *)rb,rb_size);
DumpBytes(blocks,rb,res);
if (blocks>0) break;
if (res<=0) ec++;
dr+=res;
} while ((res<=5) && (dr<0x200) && (ec<5));
//if (res>5) DumpBytes(blocks,rb,res);
2011-07-18 10:57:34 +00:00
done=false;
if (blocks==0) {
2011-07-20 03:43:41 +00:00
if (res>5)
for (int i=0;i<res-2;i++) {
if ((rb[i]==0xf2) && (rb[i+1]==0x80) && (rb[i+2]==0x00)) {
fnd=true;
2011-07-18 10:57:34 +00:00
startpos=i+9;
break;
}
}
if (!fnd) {
qDebug() << "Retrying..";
fails++;
break; // reissue the F5 and try again
2011-07-18 10:57:34 +00:00
} else {
qDebug() << "Found";
}
2011-07-18 10:57:34 +00:00
// 84 95 7c
// 0x10556 bytes..
// 0x82b blocks
2011-07-20 03:43:41 +00:00
length=0;
while (rb[startpos]!=0xf0) {
length=(length << 7) | (rb[startpos++] & 0x7f);
}
2011-07-27 09:21:53 +00:00
//length=(rb[startpos++] & 0x7f) << 14;
//length|=(rb[startpos++] & 0x7f) << 7;
//length|=(rb[startpos++] & 0x7f);
//if (!(rb[startpos]&0x80)) {
// length <<= 8;
// length|=rb[startpos++];
//}
//length=(rb[startpos] ^ 0x80)<< 7 | (rb[startpos+1]);
//startpos+=2;
//if (!(rb[startpos]&0x80)) {
// length|=(rb[startpos]&0x7f) << 14;
// startpos++;
//} else oneoff=true;
buffer=new unsigned char [length+32];
//qDebug() << length << startpos;
bytes+=res-startpos;
memcpy((char *)buffer,(char *)&rb[startpos],bytes);
} else {
qprogress->setValue((100.0/length)*bytes);
memcpy((char *)&buffer[bytes],(char *)rb,res);
bytes+=res;
}
blocks++;
2011-07-20 03:43:41 +00:00
if (res<rb_size) {
2011-07-18 10:57:34 +00:00
qDebug() << "Read "<< bytes << " bytes of " << length;
done=true;
break;
}
} while (bytes<length);
if (done) break;
if (fails>4) break;
}
if (done) {
if (oneoff) bytes--; // this is retarded..
QDateTime date=QDateTime::currentDateTimeUtc();
SessionID sid=date.toTime_t();
Session *sess=new Session(mach,sid);
qDebug() << "Read " << bytes << "Bytes";
qDebug() << "Creating session " << sid;
char pulse,spo2,lastpulse=-1,lastspo2=-1;
qint64 tt=sid-(bytes/3);
tt*=1000;
sess->set_first(tt);
EventDataType data;
unsigned i=0;
while (i<bytes) {
if (buffer[i++]!=0xf0) {
qDebug() << "Faulty PulseOx data";
continue;
}
pulse=buffer[i++] ^ 0x80;
spo2=buffer[i++];
if (pulse!=0 && pulse!=lastpulse) {
data=pulse;
2011-07-27 09:21:53 +00:00
sess->AddEvent(new Event(tt,OXI_Pulse,0,&data,1));
//qDebug() << "Pulse: " << int(pulse);
}
if (spo2 != 0 && spo2!=lastspo2) {
data=spo2;
2011-07-27 09:21:53 +00:00
sess->AddEvent(new Event(tt,OXI_SPO2,0,&data,1));
//qDebug() << "SpO2: " << int(spo2);
}
lastpulse=pulse;
lastspo2=spo2;
tt+=1000;
}
data=pulse;
2011-07-27 09:21:53 +00:00
sess->AddEvent(new Event(tt,OXI_Pulse,0,&data,1));
data=spo2;
2011-07-27 09:21:53 +00:00
sess->AddEvent(new Event(tt,OXI_SPO2,0,&data,1));
sess->summary[OXI_PulseMin]=sess->min_event_field(OXI_Pulse,0);
sess->summary[OXI_PulseMax]=sess->max_event_field(OXI_Pulse,0);
sess->summary[OXI_PulseAverage]=sess->weighted_avg_event_field(OXI_Pulse,0);
sess->summary[OXI_SPO2Min]=sess->min_event_field(OXI_SPO2,0);
sess->summary[OXI_SPO2Max]=sess->max_event_field(OXI_SPO2,0);
sess->summary[OXI_SPO2Average]=sess->weighted_avg_event_field(OXI_SPO2,0);
sess->SetChanged(true);
mach->AddSession(sess,profile);
mach->Save();
// Output Pulse & SPO2 here..
delete [] buffer;
port->write((char *)b2,3);
}
delete port;
port=NULL;
2011-07-20 03:43:41 +00:00
msgbox.hide();
qprogress->hide();
ui->SerialPortsCombo->setEnabled(true);
qstatus->setText("Ready");
2011-07-13 05:27:27 +00:00
}
2011-07-27 09:21:53 +00:00
*/