/******************************************************************************** * FaceTrackNoIR This program is a private project of some enthusiastic * * gamers from Holland, who don't like to pay much for * * head-tracking. * * * * Copyright (C) 2012 Wim Vriend (Developing) * * Ron Hendriks (Researching and Testing) * * Homepage: http://facetracknoir.sourceforge.net/home/default.htm * * * * Copyright (C) 2012 FuraX49 (HAT Tracker plugins) * * Homepage: http://hatire.sourceforge.net * * * * * * This program is free software; you can redistribute it and/or modify it * * under the terms of the GNU General Public License as published by the * * Free Software Foundation; either version 3 of the License, or (at your * * option) any later version. * * * * This program is distributed in the hope that it will be useful, but * * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * * more details. * * * * You should have received a copy of the GNU General Public License along * * with this program; if not, see . * * * ********************************************************************************/ #include #include "ftnoir_tracker_hat.h" FTNoIR_Tracker::FTNoIR_Tracker() { qDebug()<<"Tracker::HAT"; ComPort = NULL; HAT.Rot[0]=0; HAT.Rot[1]=0; HAT.Rot[2]=0; HAT.Trans[0]=0; HAT.Trans[1]=0; HAT.Trans[2]=0; // prepare & reserve QByteArray dataRead.resize(4096); dataRead.clear(); Begin.append((char) 0xAA); Begin.append((char) 0xAA); End.append((char) 0x55); End.append((char) 0x55); flDiagnostics.setFileName(QCoreApplication::applicationDirPath() + "/HATDiagnostics.txt"); settings.load_ini(); } FTNoIR_Tracker::~FTNoIR_Tracker() { qDebug()<<"Tracker::~HAT"; if (ComPort!=NULL) { if (ComPort->isOpen() ) { #ifdef OPENTRACK_API QByteArray Msg; Log("Tracker shut down"); ComPort->write(sCmdStop); if (!ComPort->waitForBytesWritten(1000)) { emit sendMsgInfo("TimeOut in writing CMD"); } else { Msg.append("\r\n"); Msg.append("SEND '"); Msg.append(sCmdStop); Msg.append("'\r\n"); } emit sendMsgInfo(Msg); #endif ComPort->close(); disconnect(ComPort,SIGNAL(readyRead()),0,0); } delete ComPort; ComPort=NULL; } } //send ZERO to Arduino bool FTNoIR_Tracker::notifyZeroed() { qDebug() << " HAT send ZEROed "; sendcmd(sCmdZero); return true; } //send RESET to Arduino void FTNoIR_Tracker::reset() { qDebug() << " HAT send RESET "; sendcmd(sCmdReset); } // Info SerialPort void FTNoIR_Tracker::SerialInfo() { QByteArray Msg; if (ComPort!=NULL) { if (ComPort->isOpen() ) { Msg.append("\r\n"); Msg.append(ComPort->portName()); Msg.append("\r\n"); Msg.append("BAUDRATE :"); Msg.append(QString::number(ComPort->baudRate())); Msg.append("\r\n"); Msg.append("DataBits :"); Msg.append(QString::number(ComPort->dataBits())); Msg.append("\r\n"); Msg.append("Parity :"); switch (ComPort->parity()) { case 0: Msg.append("No parity"); break; case 2: Msg.append("Even parity"); break; case 3: Msg.append("Odd parity"); break; case 4: Msg.append("Space parity"); break; case 5: Msg.append("Mark parity"); break; default: Msg.append("Unknown parity"); break; } Msg.append("\r\n"); Msg.append("Stop Bits :"); switch (ComPort->stopBits()) { Msg.append(QString::number(ComPort->stopBits())); case 1: Msg.append("1 stop bit."); break; case 2: Msg.append("2 stop bits."); break; case 3: Msg.append("1.5 stop bits."); break; default: Msg.append("Unknown number of stop bit."); break; } Msg.append("\r\n"); Msg.append("Flow Control :"); switch (ComPort->flowControl()) { case 0: Msg.append("No flow control"); break; case 1: Msg.append("Hardware flow control (RTS/CTS)"); break; case 2: Msg.append("Software flow control (XON/XOFF)"); break; default: Msg.append("Unknown flow control"); break; } emit sendMsgInfo(Msg); } } } //send command to Arduino void FTNoIR_Tracker::sendcmd(const QByteArray &cmd) { QByteArray Msg; if (cmd.length()>0) { if (ComPort->isOpen() ) { QString logMess; logMess.append("SEND '"); logMess.append(cmd); logMess.append("'"); Log(logMess); ComPort->write(cmd); if (!ComPort->waitForBytesWritten(1000)) { emit sendMsgInfo("TimeOut in writing CMD"); } else { Msg.append("\r\n"); Msg.append("SEND '"); Msg.append(cmd); Msg.append("'\r\n"); } #ifndef _WIN32 // WaitForReadyRead isn't working well and there are some reports of it being a win32 issue. We can live without it anyway if ( !ComPort->waitForReadyRead(1000)) { emit sendMsgInfo("TimeOut in response to CMD") ; } else { emit sendMsgInfo(Msg); } #else emit sendMsgInfo(Msg); #endif } else { emit sendMsgInfo("ComPort not open") ; } } } // return FPS void FTNoIR_Tracker::get_info( int *tps ){ *tps=frame_cnt; frame_cnt=0; } void FTNoIR_Tracker::SerialRead() { QMutexLocker lck(&mutex); dataRead+=ComPort->readAll(); } #ifndef OPENTRACK_API void FTNoIR_Tracker::Initialize( QFrame *videoframe ) { CptError=0; dataRead.clear(); frame_cnt=0; Log("INITIALISING HATIRE"); settings.load_ini(); applysettings(settings); ComPort = new QSerialPort(this); ComPort->setPortName(sSerialPortName); if (ComPort->open(QIODevice::ReadWrite ) == true) { connect(ComPort, SIGNAL(readyRead()), this, SLOT(SerialRead())); if ( ComPort->setBaudRate((QSerialPort::BaudRate)iBaudRate) && ComPort->setDataBits((QSerialPort::DataBits)iDataBits) && ComPort->setParity((QSerialPort::Parity)iParity) && ComPort->setStopBits((QSerialPort::StopBits)iStopBits) && ComPort->setFlowControl((QSerialPort::FlowControl)iFlowControl) && ComPort->clear(QSerialPort::AllDirections) && ComPort->setDataErrorPolicy(QSerialPort::IgnorePolicy) ) { // Wait init arduino sequence for (int i = 1; i <=iDelayInit; i+=50) { if (ComPort->waitForReadyRead(50)) break; } sendcmd(sCmdInit); // Wait init MPU sequence for (int i = 1; i <=iDelayStart; i+=50) { if (ComPort->waitForReadyRead(50)) break; } } else { QMessageBox::warning(0,"FaceTrackNoIR Error", ComPort->errorString(),QMessageBox::Ok,QMessageBox::NoButton); } } else { QMessageBox::warning(0,"FaceTrackNoIR Error", "Unable to open ComPort",QMessageBox::Ok,QMessageBox::NoButton); delete ComPort; ComPort = NULL; } return; } void FTNoIR_Tracker::StartTracker(HWND parent_window) { // Send START cmd to IMU sendcmd(sCmdStart); Log("Starting Tracker"); // Wait start MPU sequence for (int i = 1; i <=iDelaySeq; i+=50) { if (ComPort->waitForReadyRead(50)) break; } return; } void FTNoIR_Tracker::StopTracker( bool exit ) { QByteArray Msg; Log("Stopping tracker"); if (sCmdStop.length()>0) { if (ComPort->isOpen() ) { ComPort->write(sCmdStop); if (!ComPort->waitForBytesWritten(1000)) { emit sendMsgInfo("TimeOut in writing CMD"); } else { Msg.append("\r\n"); Msg.append("SEND '"); Msg.append(sCmdStop); Msg.append("'\r\n"); } emit sendMsgInfo(Msg); } } // OK, the thread is not stopped, doing this. That might be dangerous anyway... // if (exit || !exit) return; return; } //send CENTER to Arduino void FTNoIR_Tracker::notifyCenter() { sendcmd(sCmdCenter); } #else void FTNoIR_Tracker::start_tracker(QFrame*) { CptError=0; dataRead.clear(); frame_cnt=0; new_frame=false; settings.load_ini(); applysettings(settings); ComPort = new QSerialPort(this); ComPort->setPortName(sSerialPortName); Log("Starting Tracker"); if (ComPort->open(QIODevice::ReadWrite ) == true) { connect(ComPort, SIGNAL(readyRead()), this, SLOT(SerialRead())); Log("Port Open"); if ( ComPort->setBaudRate((QSerialPort::BaudRate)iBaudRate) && ComPort->setDataBits((QSerialPort::DataBits)iDataBits) && ComPort->setParity((QSerialPort::Parity)iParity) && ComPort->setStopBits((QSerialPort::StopBits)iStopBits) && ComPort->setFlowControl((QSerialPort::FlowControl)iFlowControl) && ComPort->clear(QSerialPort::AllDirections) && ComPort->setDataErrorPolicy(QSerialPort::IgnorePolicy) ) { Log("Port Parameters set"); qDebug() << QTime::currentTime() << " HAT OPEN on " << ComPort->portName() << ComPort->baudRate() << ComPort->dataBits() << ComPort->parity() << ComPort->stopBits() << ComPort->flowControl(); if (ComPort->flowControl() == QSerialPort::HardwareControl) { // Raise DTR Log("Raising DTR"); if (!ComPort->setDataTerminalReady(true)) Log("Couldn't set DTR"); // Raise RTS/CTS Log("Raising RTS"); if (!ComPort->setRequestToSend(true)) Log("Couldn't set RTS"); } // Wait init arduino sequence for (int i = 1; i <=iDelayInit; i+=50) { if (ComPort->waitForReadyRead(50)) break; } Log("Waiting on init"); qDebug() << QTime::currentTime() << " HAT send INIT "; sendcmd(sCmdInit); // Wait init MPU sequence for (int i = 1; i <=iDelayStart; i+=50) { if (ComPort->waitForReadyRead(50)) break; } // Send START cmd to IMU qDebug() << QTime::currentTime() << " HAT send START "; sendcmd(sCmdStart); // Wait start MPU sequence for (int i = 1; i <=iDelaySeq; i+=50) { if (ComPort->waitForReadyRead(50)) break; } Log("Port setup, waiting for HAT frames to process"); qDebug() << QTime::currentTime() << " HAT wait MPU "; } else { QMessageBox::warning(0,"FaceTrackNoIR Error", ComPort->errorString(),QMessageBox::Ok,QMessageBox::NoButton); } } else { QMessageBox::warning(0,"FaceTrackNoIR Error", "Unable to open ComPort: " + ComPort->errorString(), QMessageBox::Ok,QMessageBox::NoButton); delete ComPort; ComPort = NULL; } return; } //send CENTER to Arduino void FTNoIR_Tracker::center() { qDebug() << " HAT send CENTER "; Log("Sending Centre Command"); sendcmd(sCmdCenter); } //Return speed FPS sketch Arduino int FTNoIR_Tracker::preferredHz() { qDebug() << " HAT return Preferred FPS " << iFpsArduino; return iFpsArduino; } #endif // // Return 6DOF info // #ifdef OPENTRACK_API void FTNoIR_Tracker::data(double *data) #else bool FTNoIR_Tracker::GiveHeadPoseData(THeadPoseData *data) #endif { QMutexLocker lck(&mutex); while (dataRead.length()>=30) { Log(dataRead.toHex()); if ((dataRead.startsWith(Begin) && ( dataRead.mid(28,2)==End )) ) { // .Begin==0xAAAA .End==0x5555 QDataStream datastream(dataRead.left(30)); if (bBigEndian) datastream.setByteOrder(QDataStream::BigEndian ); else datastream.setByteOrder(QDataStream::LittleEndian ); datastream>>ArduinoData; frame_cnt++; if (ArduinoData.Code <= 1000) { HAT=ArduinoData; new_frame=true; } else { emit sendMsgInfo(dataRead.mid(4,24)) ; } dataRead.remove(0,30); } else { // resynchro trame int index = dataRead.indexOf(Begin); if (index==-1) { index=dataRead.length(); } emit sendMsgInfo(dataRead.mid(0,index)) ; dataRead.remove(0,index); CptError++; qDebug() << QTime::currentTime() << " HAT Resync-Frame, counter " << CptError; } } if (CptError>50) { emit sendMsgInfo("Can't find HAT frame") ; CptError=0; #ifndef OPENTRACK_API return false; #endif } // Need to handle this differently in opentrack as opposed to tracknoir //if (new_frame) { #ifdef OPENTRACK_API // in open track always populate the data, it seems opentrack always gives us a zeroed data structure to populate with pose data. // if we have no new data, we don't populate it and so 0 pose gets handed back which is wrong. By always running the code below, if we // have no new data, we will just give it the previous pose data which is the best thing we can do really. if(1){ if (bEnableYaw) { if (bInvertYaw ) data[Yaw] = HAT.Rot[iYawAxe] * -1.0f; else data[Yaw] = HAT.Rot[iYawAxe]; } else data[Yaw] =0; if (bEnablePitch) { if (bInvertPitch) data[Pitch] = HAT.Rot[iPitchAxe] * -1.0f; else data[Pitch] = HAT.Rot[iPitchAxe]; } else data[Pitch] = 0; if (bEnableRoll) { if (bInvertRoll) data[Roll] = HAT.Rot[iRollAxe] * -1.0f; else data[Roll] = HAT.Rot[iRollAxe]; } else data[Roll] =0; if (bEnableX) { if (bInvertX) data[TX] = HAT.Trans[iXAxe]* -1.0f; else data[TX] = HAT.Trans[iXAxe]; } else data[TX] =0; if (bEnableY) { if (bInvertY) data[TY] = HAT.Trans[iYAxe]* -1.0f; else data[TY] = HAT.Trans[iYAxe]; } else data[TY] =0; if (bEnableZ) { if (bInvertZ) data[TZ] = HAT.Trans[iZAxe]* -1.0f; else data[TZ] = HAT.Trans[iZAxe]; } else data[TZ] =0; #else if (new_frame) { // treat frame handling as it was for TrackNoIR. if (bEnableYaw) { if (bInvertYaw ) data->yaw = (double) HAT.Rot[iYawAxe] * -1.0f; else data->yaw = (double) HAT.Rot[iYawAxe]; } if (bEnablePitch) { if (bInvertPitch)data->pitch = (double) HAT.Rot[iPitchAxe] * -1.0f; else data->pitch = (double) HAT.Rot[iPitchAxe]; } if (bEnableRoll) { if (bInvertRoll) data->roll = (double) HAT.Rot[iRollAxe] * -1.0f; else data->roll = (double) HAT.Rot[iRollAxe]; } if (bEnableX) { if (bInvertX) data->x = (double) HAT.Trans[iXAxe]* -1.0f; else data->x = (double) HAT.Trans[iXAxe]; } if (bEnableY) { if (bInvertY) data->y = (double) HAT.Trans[iYAxe]* -1.0f; else data->y = (double) HAT.Trans[iYAxe]; } if (bEnableZ) { if (bInvertZ) data->z = (double) HAT.Trans[iZAxe]* -1.0f; else data->z = (double) HAT.Trans[iZAxe]; } return true; #endif new_frame=false; // For debug //data->x=dataRead.length(); //data->y=CptError; } } // // Apply modification Settings // void FTNoIR_Tracker::applysettings(const TrackerSettings& settings){ QMutexLocker lck(&mutex); sSerialPortName= settings.SerialPortName; bEnableRoll = settings.EnableRoll; bEnablePitch = settings.EnablePitch; bEnableYaw = settings.EnableYaw; bEnableX = settings.EnableX; bEnableY = settings.EnableY; bEnableZ = settings.EnableZ; bInvertRoll = settings.InvertRoll; bInvertPitch = settings.InvertPitch; bInvertYaw = settings.InvertYaw; bInvertX = settings.InvertX; bInvertY = settings.InvertY; bInvertZ = settings.InvertZ; bEnableLogging = settings.EnableLogging; iRollAxe= settings.RollAxe; iPitchAxe= settings.PitchAxe; iYawAxe= settings.YawAxe; iXAxe= settings.XAxe; iYAxe= settings.YAxe; iZAxe= settings.ZAxe; iBaudRate=settings.pBaudRate; iDataBits=settings.pDataBits; iParity=settings.pParity; iStopBits=settings.pStopBits; iFlowControl=settings.pFlowControl; sCmdStart= settings.CmdStart.toLatin1(); sCmdStop= settings.CmdStop.toLatin1(); sCmdInit= settings.CmdInit.toLatin1(); sCmdReset= settings.CmdReset.toLatin1(); sCmdCenter= settings.CmdCenter.toLatin1(); sCmdZero= settings.CmdZero.toLatin1(); iDelayInit=settings.DelayInit; iDelayStart=settings.DelayStart; iDelaySeq=settings.DelaySeq; bBigEndian=settings.BigEndian; #ifdef OPENTRACK_API iFpsArduino=settings.FPSArduino; #endif } void FTNoIR_Tracker::Log(QString message) { // Drop out immediately if logging is off. Yes, there is still some overhead because of passing strings around for no reason. // that's unfortunate and I'll monitor the impact and see if it needs a more involved fix. if (!bEnableLogging) return; QString logMessage; if (flDiagnostics.open(QIODevice::ReadWrite | QIODevice::Append)) { QTextStream out(&flDiagnostics); QString milliSeconds; milliSeconds = QString("%1").arg(QTime::currentTime().msec(), 3, 10, QChar('0')); // We have a file out << QTime::currentTime().toString() << "." << milliSeconds << ": " << message << "\r\n"; flDiagnostics.close(); } } //////////////////////////////////////////////////////////////////////////////// // Factory function that creates instances if the Tracker object. // Export both decorated and undecorated names. // GetTracker - Undecorated name, which can be easily used with GetProcAddress // Win32 API function. // _GetTracker@0 - Common name decoration for __stdcall functions in C language. //////////////////////////////////////////////////////////////////////////////// #ifdef OPENTRACK_API extern "C" OPENTRACK_EXPORT ITracker* GetConstructor() #else #pragma comment(linker, "/export:GetTracker=_GetTracker@0") FTNOIR_TRACKER_BASE_EXPORT ITrackerPtr __stdcall GetTracker() #endif { return new FTNoIR_Tracker; }