2010-06-23 02:33:53 +00:00
/**
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
2010-07-16 18:07:13 +00:00
* @ file uavobjectfield . cpp
2010-06-23 02:33:53 +00:00
* @ author The OpenPilot Team , http : //www.openpilot.org Copyright (C) 2010.
* @ see The GNU Public License ( GPL ) Version 3
2010-07-16 17:32:18 +00:00
* @ addtogroup GCSPlugins GCS Plugins
2010-06-23 02:33:53 +00:00
* @ {
2010-07-16 17:32:18 +00:00
* @ addtogroup UAVObjectsPlugin UAVObjects Plugin
* @ {
2010-07-16 18:07:13 +00:00
* @ brief The UAVUObjects GCS plugin
2010-06-23 02:33:53 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-07-16 17:32:18 +00:00
/*
2012-02-05 14:41:23 +00:00
* 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
2010-06-23 02:33:53 +00:00
* ( at your option ) any later version .
2012-02-05 14:41:23 +00:00
*
* 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
2010-06-23 02:33:53 +00:00
* for more details .
2012-02-05 14:41:23 +00:00
*
* You should have received a copy of the GNU General Public License along
* with this program ; if not , write to the Free Software Foundation , Inc . ,
2010-06-23 02:33:53 +00:00
* 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "uavobjectfield.h"
# include <QtEndian>
2010-09-21 16:36:04 +00:00
# include <QDebug>
2010-06-23 02:33:53 +00:00
2012-02-03 20:29:28 +00:00
UAVObjectField : : UAVObjectField ( const QString & name , const QString & units , FieldType type , quint32 numElements , const QStringList & options , const QString & limits )
2010-06-23 02:33:53 +00:00
{
QStringList elementNames ;
// Set element names
for ( quint32 n = 0 ; n < numElements ; + + n )
{
elementNames . append ( QString ( " %1 " ) . arg ( n ) ) ;
}
// Initialize
2012-02-03 20:29:28 +00:00
constructorInitialize ( name , units , type , elementNames , options , limits ) ;
2010-06-23 02:33:53 +00:00
}
2012-02-03 20:29:28 +00:00
UAVObjectField : : UAVObjectField ( const QString & name , const QString & units , FieldType type , const QStringList & elementNames , const QStringList & options , const QString & limits )
2010-06-23 02:33:53 +00:00
{
2012-02-03 20:29:28 +00:00
constructorInitialize ( name , units , type , elementNames , options , limits ) ;
2010-06-23 02:33:53 +00:00
}
2012-02-05 14:41:23 +00:00
void UAVObjectField : : constructorInitialize ( const QString & name , const QString & units , FieldType type , const QStringList & elementNames , const QStringList & options , const QString & limits )
2010-06-23 02:33:53 +00:00
{
// Copy params
this - > name = name ;
this - > units = units ;
this - > type = type ;
this - > options = options ;
this - > numElements = elementNames . length ( ) ;
this - > offset = 0 ;
this - > data = NULL ;
this - > obj = NULL ;
this - > elementNames = elementNames ;
// Set field size
switch ( type )
{
2012-02-05 14:41:23 +00:00
case INT8 :
numBytesPerElement = sizeof ( qint8 ) ;
break ;
case INT16 :
numBytesPerElement = sizeof ( qint16 ) ;
break ;
case INT32 :
numBytesPerElement = sizeof ( qint32 ) ;
break ;
case UINT8 :
numBytesPerElement = sizeof ( quint8 ) ;
break ;
case UINT16 :
numBytesPerElement = sizeof ( quint16 ) ;
break ;
case UINT32 :
numBytesPerElement = sizeof ( quint32 ) ;
break ;
case FLOAT32 :
numBytesPerElement = sizeof ( quint32 ) ;
break ;
case ENUM :
numBytesPerElement = sizeof ( quint8 ) ;
break ;
2012-05-15 18:22:36 +02:00
case BITFIELD :
numBytesPerElement = sizeof ( quint8 ) ;
2012-05-15 19:40:58 +02:00
this - > options = QStringList ( ) < < tr ( " 0 " ) < < tr ( " 1 " ) ;
2012-05-15 18:22:36 +02:00
break ;
2012-02-05 14:41:23 +00:00
case STRING :
numBytesPerElement = sizeof ( quint8 ) ;
break ;
default :
numBytesPerElement = 0 ;
2010-06-23 02:33:53 +00:00
}
2012-02-05 14:41:23 +00:00
limitsInitialize ( limits ) ;
2010-06-23 02:33:53 +00:00
}
2012-02-05 14:41:23 +00:00
void UAVObjectField : : limitsInitialize ( const QString & limits )
2012-02-03 20:29:28 +00:00
{
/// format
/// (TY)->type (EQ-equal;NE-not equal;BE-between;BI-bigger;SM-smaller)
/// (VALX)->value
/// %TY:VAL1:VAL2:VAL3,%TY,VAL1,VAL2,VAL3
/// example: first element bigger than 3 and second element inside [2.3,5]
2012-02-11 18:35:36 +00:00
/// "%BI:3,%BE:2.3:5"
2012-02-05 14:41:23 +00:00
if ( limits . isEmpty ( ) )
return ;
2012-02-03 20:29:28 +00:00
QStringList stringPerElement = limits . split ( " , " ) ;
2012-02-10 16:34:07 +00:00
quint32 index = 0 ;
2012-02-03 20:29:28 +00:00
foreach ( QString str , stringPerElement ) {
2012-08-02 23:56:02 +01:00
QStringList ruleList = str . split ( " ; " ) ;
QList < LimitStruct > limitList ;
foreach ( QString rule , ruleList )
2012-02-03 20:29:28 +00:00
{
2012-08-02 23:56:02 +01:00
QString _str = rule . trimmed ( ) ;
if ( _str . isEmpty ( ) )
continue ;
QStringList valuesPerElement = _str . split ( " : " ) ;
LimitStruct lstruc ;
bool b1 = valuesPerElement . at ( 0 ) . startsWith ( " % " ) ;
bool b2 = ( int ) ( index ) < ( int ) numElements ;
bool b3 = valuesPerElement . at ( 0 ) . size ( ) = = 3 ;
bool auxb ;
valuesPerElement . at ( 0 ) . mid ( 1 , 4 ) . toInt ( & auxb , 16 ) ;
bool b4 = ( ( valuesPerElement . at ( 0 ) . size ( ) ) = = 7 & & auxb ) ;
if ( b1 & & b2 & & ( b3 | | b4 ) )
2012-02-03 20:29:28 +00:00
{
2012-08-02 23:56:02 +01:00
if ( b4 )
lstruc . board = valuesPerElement . at ( 0 ) . mid ( 1 , 4 ) . toInt ( & auxb , 16 ) ;
else
lstruc . board = 0 ;
if ( valuesPerElement . at ( 0 ) . right ( 2 ) = = " EQ " )
lstruc . type = EQUAL ;
else if ( valuesPerElement . at ( 0 ) . right ( 2 ) = = " NE " )
lstruc . type = NOT_EQUAL ;
else if ( valuesPerElement . at ( 0 ) . right ( 2 ) = = " BE " )
lstruc . type = BETWEEN ;
else if ( valuesPerElement . at ( 0 ) . right ( 2 ) = = " BI " )
lstruc . type = BIGGER ;
else if ( valuesPerElement . at ( 0 ) . right ( 2 ) = = " SM " )
lstruc . type = SMALLER ;
else
qDebug ( ) < < " limits parsing failed (invalid property) on UAVObjectField " < < name ;
valuesPerElement . removeAt ( 0 ) ;
foreach ( QString _value , valuesPerElement )
2012-02-05 14:41:23 +00:00
{
2012-08-02 23:56:02 +01:00
QString value = _value . trimmed ( ) ;
switch ( type )
{
case UINT8 :
case UINT16 :
case UINT32 :
case BITFIELD :
lstruc . values . append ( ( quint32 ) value . toULong ( ) ) ;
break ;
case INT8 :
case INT16 :
case INT32 :
lstruc . values . append ( ( qint32 ) value . toLong ( ) ) ;
break ;
case FLOAT32 :
lstruc . values . append ( ( float ) value . toFloat ( ) ) ;
break ;
case ENUM :
lstruc . values . append ( ( QString ) value ) ;
break ;
case STRING :
lstruc . values . append ( ( QString ) value ) ;
break ;
default :
lstruc . values . append ( QVariant ( ) ) ;
}
2012-02-05 14:41:23 +00:00
}
2012-08-02 23:56:02 +01:00
limitList . append ( lstruc ) ;
}
else
{
if ( ! valuesPerElement . at ( 0 ) . isEmpty ( ) & & ! b1 )
qDebug ( ) < < " limits parsing failed (property doesn't start with %) on UAVObjectField " < < name ;
else if ( ! b2 )
qDebug ( ) < < " limits parsing failed (index>numelements) on UAVObjectField " < < name < < " index " < < index < < " numElements " < < numElements ;
else if ( ! b3 | | ! b4 )
qDebug ( ) < < " limits parsing failed limit not starting with %XX or %YYYYXX where XX is the limit type and YYYY is the board type on UAVObjectField " < < name ;
2012-02-03 20:29:28 +00:00
}
}
2012-08-02 23:56:02 +01:00
elementLimits . insert ( index , limitList ) ;
+ + index ;
}
foreach ( QList < LimitStruct > limitList , elementLimits )
{
foreach ( LimitStruct limit , limitList )
2012-02-06 17:33:54 +00:00
{
2012-08-02 23:56:02 +01:00
qDebug ( ) < < " Limit type " < < limit . type < < " for board " < < limit . board < < " for field " < < getName ( ) ;
foreach ( QVariant var , limit . values )
{
qDebug ( ) < < " value " < < var ;
}
2012-02-06 17:33:54 +00:00
}
2012-02-03 20:29:28 +00:00
}
}
2012-08-01 14:42:21 +01:00
bool UAVObjectField : : isWithinLimits ( QVariant var , quint32 index , int board )
2012-02-05 14:41:23 +00:00
{
if ( ! elementLimits . keys ( ) . contains ( index ) )
return true ;
2012-08-02 23:56:02 +01:00
foreach ( LimitStruct struc , elementLimits . value ( index ) )
2012-02-05 14:41:23 +00:00
{
2012-08-02 23:56:02 +01:00
if ( ( struc . board ! = board ) & & board ! = 0 & & struc . board ! = 0 )
continue ;
switch ( struc . type )
2012-02-05 14:41:23 +00:00
{
2012-08-02 23:56:02 +01:00
case EQUAL :
switch ( type )
{
case INT8 :
case INT16 :
case INT32 :
foreach ( QVariant vars , struc . values ) {
if ( var . toInt ( ) = = vars . toInt ( ) )
return true ;
}
return false ;
break ;
case UINT8 :
case UINT16 :
case UINT32 :
case BITFIELD :
foreach ( QVariant vars , struc . values ) {
if ( var . toUInt ( ) = = vars . toUInt ( ) )
return true ;
}
return false ;
break ;
case ENUM :
case STRING :
foreach ( QVariant vars , struc . values ) {
if ( var . toString ( ) = = vars . toString ( ) )
return true ;
}
return false ;
break ;
case FLOAT32 :
foreach ( QVariant vars , struc . values ) {
if ( var . toFloat ( ) = = vars . toFloat ( ) )
return true ;
}
return false ;
break ;
default :
return true ;
2012-02-05 14:41:23 +00:00
}
break ;
2012-08-02 23:56:02 +01:00
case NOT_EQUAL :
switch ( type )
{
case INT8 :
case INT16 :
case INT32 :
foreach ( QVariant vars , struc . values ) {
if ( var . toInt ( ) = = vars . toInt ( ) )
return false ;
}
return true ;
break ;
case UINT8 :
case UINT16 :
case UINT32 :
case BITFIELD :
foreach ( QVariant vars , struc . values ) {
if ( var . toUInt ( ) = = vars . toUInt ( ) )
return false ;
}
return true ;
break ;
case ENUM :
case STRING :
foreach ( QVariant vars , struc . values ) {
if ( var . toString ( ) = = vars . toString ( ) )
return false ;
}
return true ;
break ;
case FLOAT32 :
foreach ( QVariant vars , struc . values ) {
if ( var . toFloat ( ) = = vars . toFloat ( ) )
return false ;
}
return true ;
break ;
default :
return true ;
2012-02-05 14:41:23 +00:00
}
break ;
2012-08-02 23:56:02 +01:00
case BETWEEN :
if ( struc . values . length ( ) < 2 )
{
qDebug ( ) < < __FUNCTION__ < < " between limit with less than 1 pair, aborting; field: " < < name ;
return true ;
2012-02-05 14:41:23 +00:00
}
2012-08-02 23:56:02 +01:00
if ( struc . values . length ( ) > 2 )
qDebug ( ) < < __FUNCTION__ < < " between limit with more than 1 pair, using first; field " < < name ;
switch ( type )
{
case INT8 :
case INT16 :
case INT32 :
if ( ! ( var . toInt ( ) > = struc . values . at ( 0 ) . toInt ( ) & & var . toInt ( ) < = struc . values . at ( 1 ) . toInt ( ) ) )
return false ;
return true ;
break ;
case UINT8 :
case UINT16 :
case UINT32 :
case BITFIELD :
if ( ! ( var . toUInt ( ) > = struc . values . at ( 0 ) . toUInt ( ) & & var . toUInt ( ) < = struc . values . at ( 1 ) . toUInt ( ) ) )
return false ;
return true ;
break ;
case ENUM :
if ( ! ( options . indexOf ( var . toString ( ) ) > = options . indexOf ( struc . values . at ( 0 ) . toString ( ) ) & & options . indexOf ( var . toString ( ) ) < = options . indexOf ( struc . values . at ( 1 ) . toString ( ) ) ) )
return false ;
return true ;
break ;
case STRING :
return true ;
break ;
case FLOAT32 :
if ( ! ( var . toFloat ( ) > = struc . values . at ( 0 ) . toFloat ( ) & & var . toFloat ( ) < = struc . values . at ( 1 ) . toFloat ( ) ) )
return false ;
return true ;
break ;
default :
return true ;
2012-02-05 14:41:23 +00:00
}
break ;
2012-08-02 23:56:02 +01:00
case BIGGER :
if ( struc . values . length ( ) < 1 )
{
qDebug ( ) < < __FUNCTION__ < < " BIGGER limit with less than 1 value, aborting; field: " < < name ;
return true ;
2012-02-05 14:41:23 +00:00
}
2012-08-02 23:56:02 +01:00
if ( struc . values . length ( ) > 1 )
qDebug ( ) < < __FUNCTION__ < < " BIGGER limit with more than 1 value, using first; field " < < name ;
switch ( type )
{
case INT8 :
case INT16 :
case INT32 :
if ( ! ( var . toInt ( ) > = struc . values . at ( 0 ) . toInt ( ) ) )
return false ;
return true ;
break ;
case UINT8 :
case UINT16 :
case UINT32 :
case BITFIELD :
if ( ! ( var . toUInt ( ) > = struc . values . at ( 0 ) . toUInt ( ) ) )
return false ;
return true ;
break ;
case ENUM :
if ( ! ( options . indexOf ( var . toString ( ) ) > = options . indexOf ( struc . values . at ( 0 ) . toString ( ) ) ) )
return false ;
return true ;
break ;
case STRING :
return true ;
break ;
case FLOAT32 :
if ( ! ( var . toFloat ( ) > = struc . values . at ( 0 ) . toFloat ( ) ) )
return false ;
return true ;
break ;
default :
return true ;
2012-02-05 14:41:23 +00:00
}
break ;
2012-08-02 23:56:02 +01:00
case SMALLER :
switch ( type )
{
case INT8 :
case INT16 :
case INT32 :
if ( ! ( var . toInt ( ) < = struc . values . at ( 0 ) . toInt ( ) ) )
return false ;
return true ;
break ;
case UINT8 :
case UINT16 :
case UINT32 :
case BITFIELD :
if ( ! ( var . toUInt ( ) < = struc . values . at ( 0 ) . toUInt ( ) ) )
return false ;
return true ;
break ;
case ENUM :
if ( ! ( options . indexOf ( var . toString ( ) ) < = options . indexOf ( struc . values . at ( 0 ) . toString ( ) ) ) )
return false ;
return true ;
break ;
case STRING :
return true ;
break ;
case FLOAT32 :
if ( ! ( var . toFloat ( ) < = struc . values . at ( 0 ) . toFloat ( ) ) )
return false ;
return true ;
break ;
default :
return true ;
2012-02-05 14:41:23 +00:00
}
}
}
return true ;
}
2012-02-03 20:29:28 +00:00
2012-08-01 14:42:21 +01:00
QVariant UAVObjectField : : getMaxLimit ( quint32 index , int board )
2012-02-05 14:41:23 +00:00
{
if ( ! elementLimits . keys ( ) . contains ( index ) )
return QVariant ( ) ;
2012-08-02 23:56:02 +01:00
foreach ( LimitStruct struc , elementLimits . value ( index ) )
2012-02-05 14:41:23 +00:00
{
2012-08-02 23:56:02 +01:00
if ( ( struc . board ! = board ) & & board ! = 0 & & struc . board ! = 0 )
continue ;
switch ( struc . type )
{
case EQUAL :
case NOT_EQUAL :
case BIGGER :
return QVariant ( ) ;
break ;
break ;
case BETWEEN :
return struc . values . at ( 1 ) ;
break ;
case SMALLER :
return struc . values . at ( 0 ) ;
break ;
default :
return QVariant ( ) ;
break ;
}
2012-02-05 14:41:23 +00:00
}
2012-08-02 23:56:02 +01:00
return QVariant ( ) ;
2012-02-05 14:41:23 +00:00
}
2012-08-01 14:42:21 +01:00
QVariant UAVObjectField : : getMinLimit ( quint32 index , int board )
2012-02-05 14:41:23 +00:00
{
if ( ! elementLimits . keys ( ) . contains ( index ) )
return QVariant ( ) ;
2012-08-02 23:56:02 +01:00
foreach ( LimitStruct struc , elementLimits . value ( index ) )
2012-02-05 14:41:23 +00:00
{
2012-08-02 23:56:02 +01:00
if ( ( struc . board ! = board ) & & board ! = 0 & & struc . board ! = 0 )
return QVariant ( ) ;
switch ( struc . type )
{
case EQUAL :
case NOT_EQUAL :
case SMALLER :
return QVariant ( ) ;
break ;
break ;
case BETWEEN :
return struc . values . at ( 0 ) ;
break ;
case BIGGER :
return struc . values . at ( 0 ) ;
break ;
default :
return QVariant ( ) ;
break ;
}
2012-02-05 14:41:23 +00:00
}
2012-08-02 23:56:02 +01:00
return QVariant ( ) ;
2012-02-05 14:41:23 +00:00
}
2010-06-23 02:33:53 +00:00
void UAVObjectField : : initialize ( quint8 * data , quint32 dataOffset , UAVObject * obj )
{
this - > data = data ;
this - > offset = dataOffset ;
this - > obj = obj ;
clear ( ) ;
}
UAVObjectField : : FieldType UAVObjectField : : getType ( )
{
return type ;
}
2011-02-23 20:27:05 +00:00
QString UAVObjectField : : getTypeAsString ( )
{
switch ( type )
{
2012-02-05 14:41:23 +00:00
case UAVObjectField : : INT8 :
return " int8 " ;
case UAVObjectField : : INT16 :
return " int16 " ;
case UAVObjectField : : INT32 :
return " int32 " ;
case UAVObjectField : : UINT8 :
return " uint8 " ;
case UAVObjectField : : UINT16 :
return " uint16 " ;
case UAVObjectField : : UINT32 :
return " uint32 " ;
case UAVObjectField : : FLOAT32 :
return " float32 " ;
case UAVObjectField : : ENUM :
return " enum " ;
2012-05-15 19:01:32 +02:00
case UAVObjectField : : BITFIELD :
return " bitfield " ;
2012-02-05 14:41:23 +00:00
case UAVObjectField : : STRING :
return " string " ;
default :
return " " ;
2011-02-23 20:27:05 +00:00
}
}
2010-06-23 02:33:53 +00:00
QStringList UAVObjectField : : getElementNames ( )
{
return elementNames ;
}
UAVObject * UAVObjectField : : getObject ( )
{
return obj ;
}
void UAVObjectField : : clear ( )
{
QMutexLocker locker ( obj - > getMutex ( ) ) ;
2012-05-15 18:22:36 +02:00
switch ( type )
{
case BITFIELD :
2012-05-17 00:25:30 +02:00
memset ( & data [ offset ] , 0 , numBytesPerElement * ( ( quint32 ) ( 1 + ( numElements - 1 ) / 8 ) ) ) ;
2012-05-15 18:22:36 +02:00
break ;
default :
memset ( & data [ offset ] , 0 , numBytesPerElement * numElements ) ;
break ;
}
2010-06-23 02:33:53 +00:00
}
QString UAVObjectField : : getName ( )
{
return name ;
}
QString UAVObjectField : : getUnits ( )
{
return units ;
}
QStringList UAVObjectField : : getOptions ( )
{
return options ;
}
quint32 UAVObjectField : : getNumElements ( )
{
return numElements ;
}
quint32 UAVObjectField : : getDataOffset ( )
{
return offset ;
}
quint32 UAVObjectField : : getNumBytes ( )
{
2012-05-15 18:22:36 +02:00
switch ( type )
{
case BITFIELD :
2012-05-17 00:17:53 +02:00
return numBytesPerElement * ( ( quint32 ) ( 1 + ( numElements - 1 ) / 8 ) ) ;
2012-05-15 18:22:36 +02:00
break ;
default :
return numBytesPerElement * numElements ;
break ;
}
2010-06-23 02:33:53 +00:00
}
QString UAVObjectField : : toString ( )
{
QString sout ;
sout . append ( QString ( " %1: [ " ) . arg ( name ) ) ;
for ( unsigned int n = 0 ; n < numElements ; + + n )
{
sout . append ( QString ( " %1 " ) . arg ( getDouble ( n ) ) ) ;
}
sout . append ( QString ( " ] %1 \n " ) . arg ( units ) ) ;
return sout ;
}
qint32 UAVObjectField : : pack ( quint8 * dataOut )
{
QMutexLocker locker ( obj - > getMutex ( ) ) ;
// Pack each element in output buffer
switch ( type )
{
2012-02-05 14:41:23 +00:00
case INT8 :
memcpy ( dataOut , & data [ offset ] , numElements ) ;
break ;
case INT16 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
qint16 value ;
memcpy ( & value , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
qToLittleEndian < qint16 > ( value , & dataOut [ numBytesPerElement * index ] ) ;
}
break ;
case INT32 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
qint32 value ;
memcpy ( & value , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
qToLittleEndian < qint32 > ( value , & dataOut [ numBytesPerElement * index ] ) ;
}
break ;
case UINT8 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
dataOut [ numBytesPerElement * index ] = data [ offset + numBytesPerElement * index ] ;
}
break ;
case UINT16 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
quint16 value ;
memcpy ( & value , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
qToLittleEndian < quint16 > ( value , & dataOut [ numBytesPerElement * index ] ) ;
}
break ;
case UINT32 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
quint32 value ;
memcpy ( & value , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
qToLittleEndian < quint32 > ( value , & dataOut [ numBytesPerElement * index ] ) ;
}
break ;
case FLOAT32 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
quint32 value ;
memcpy ( & value , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
qToLittleEndian < quint32 > ( value , & dataOut [ numBytesPerElement * index ] ) ;
}
break ;
case ENUM :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
dataOut [ numBytesPerElement * index ] = data [ offset + numBytesPerElement * index ] ;
}
break ;
2012-05-15 18:22:36 +02:00
case BITFIELD :
2012-05-17 00:17:53 +02:00
for ( quint32 index = 0 ; index < ( quint32 ) ( 1 + ( numElements - 1 ) / 8 ) ; + + index )
2012-05-15 18:22:36 +02:00
{
dataOut [ numBytesPerElement * index ] = data [ offset + numBytesPerElement * index ] ;
}
break ;
2012-02-05 14:41:23 +00:00
case STRING :
memcpy ( dataOut , & data [ offset ] , numElements ) ;
break ;
2010-06-23 02:33:53 +00:00
}
// Done
return getNumBytes ( ) ;
}
qint32 UAVObjectField : : unpack ( const quint8 * dataIn )
{
QMutexLocker locker ( obj - > getMutex ( ) ) ;
// Unpack each element from input buffer
switch ( type )
{
2012-02-05 14:41:23 +00:00
case INT8 :
memcpy ( & data [ offset ] , dataIn , numElements ) ;
break ;
case INT16 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
qint16 value ;
value = qFromLittleEndian < qint16 > ( & dataIn [ numBytesPerElement * index ] ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & value , numBytesPerElement ) ;
}
break ;
case INT32 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
qint32 value ;
value = qFromLittleEndian < qint32 > ( & dataIn [ numBytesPerElement * index ] ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & value , numBytesPerElement ) ;
}
break ;
case UINT8 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
data [ offset + numBytesPerElement * index ] = dataIn [ numBytesPerElement * index ] ;
}
break ;
case UINT16 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
quint16 value ;
value = qFromLittleEndian < quint16 > ( & dataIn [ numBytesPerElement * index ] ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & value , numBytesPerElement ) ;
}
break ;
case UINT32 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
quint32 value ;
value = qFromLittleEndian < quint32 > ( & dataIn [ numBytesPerElement * index ] ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & value , numBytesPerElement ) ;
}
break ;
case FLOAT32 :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
quint32 value ;
value = qFromLittleEndian < quint32 > ( & dataIn [ numBytesPerElement * index ] ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & value , numBytesPerElement ) ;
}
break ;
case ENUM :
for ( quint32 index = 0 ; index < numElements ; + + index )
{
data [ offset + numBytesPerElement * index ] = dataIn [ numBytesPerElement * index ] ;
}
break ;
2012-05-15 18:22:36 +02:00
case BITFIELD :
2012-05-17 00:17:53 +02:00
for ( quint32 index = 0 ; index < ( quint32 ) ( 1 + ( numElements - 1 ) / 8 ) ; + + index )
2012-05-15 18:22:36 +02:00
{
data [ offset + numBytesPerElement * index ] = dataIn [ numBytesPerElement * index ] ;
}
break ;
2012-02-05 14:41:23 +00:00
case STRING :
memcpy ( & data [ offset ] , dataIn , numElements ) ;
break ;
2010-06-23 02:33:53 +00:00
}
// Done
return getNumBytes ( ) ;
}
bool UAVObjectField : : isNumeric ( )
{
switch ( type )
{
2012-02-05 14:41:23 +00:00
case INT8 :
return true ;
break ;
case INT16 :
return true ;
break ;
case INT32 :
return true ;
break ;
case UINT8 :
return true ;
break ;
case UINT16 :
return true ;
break ;
case UINT32 :
return true ;
break ;
case FLOAT32 :
return true ;
break ;
case ENUM :
return false ;
break ;
2012-05-15 18:22:36 +02:00
case BITFIELD :
2012-05-15 19:01:32 +02:00
return true ;
2012-05-15 18:22:36 +02:00
break ;
2012-02-05 14:41:23 +00:00
case STRING :
return false ;
break ;
default :
return false ;
2010-06-23 02:33:53 +00:00
}
}
bool UAVObjectField : : isText ( )
{
switch ( type )
{
2012-02-05 14:41:23 +00:00
case INT8 :
return false ;
break ;
case INT16 :
return false ;
break ;
case INT32 :
return false ;
break ;
case UINT8 :
return false ;
break ;
case UINT16 :
return false ;
break ;
case UINT32 :
return false ;
break ;
case FLOAT32 :
return false ;
break ;
case ENUM :
return true ;
break ;
2012-05-15 18:22:36 +02:00
case BITFIELD :
2012-05-15 19:01:32 +02:00
return false ;
2012-05-15 18:22:36 +02:00
break ;
2012-02-05 14:41:23 +00:00
case STRING :
return true ;
break ;
default :
return false ;
}
}
QVariant UAVObjectField : : getValue ( quint32 index )
{
QMutexLocker locker ( obj - > getMutex ( ) ) ;
// Check that index is not out of bounds
if ( index > = numElements )
{
return QVariant ( ) ;
}
// Get value
switch ( type )
{
case INT8 :
{
qint8 tmpint8 ;
memcpy ( & tmpint8 , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpint8 ) ;
break ;
}
case INT16 :
{
qint16 tmpint16 ;
memcpy ( & tmpint16 , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpint16 ) ;
break ;
}
case INT32 :
{
qint32 tmpint32 ;
memcpy ( & tmpint32 , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpint32 ) ;
break ;
}
case UINT8 :
{
quint8 tmpuint8 ;
memcpy ( & tmpuint8 , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpuint8 ) ;
break ;
}
case UINT16 :
{
quint16 tmpuint16 ;
memcpy ( & tmpuint16 , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpuint16 ) ;
break ;
}
case UINT32 :
{
quint32 tmpuint32 ;
memcpy ( & tmpuint32 , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpuint32 ) ;
break ;
}
case FLOAT32 :
{
float tmpfloat ;
memcpy ( & tmpfloat , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
return QVariant ( tmpfloat ) ;
break ;
}
case ENUM :
{
quint8 tmpenum ;
memcpy ( & tmpenum , & data [ offset + numBytesPerElement * index ] , numBytesPerElement ) ;
// Q_ASSERT((tmpenum < options.length()) && (tmpenum >= 0)); // catch bad enum settings
if ( tmpenum > = options . length ( ) ) {
qDebug ( ) < < " Invalid value for " < < name ;
return QVariant ( QString ( " Bad Value " ) ) ;
}
return QVariant ( options [ tmpenum ] ) ;
break ;
}
2012-05-15 18:22:36 +02:00
case BITFIELD :
{
quint8 tmpbitfield ;
memcpy ( & tmpbitfield , & data [ offset + numBytesPerElement * ( ( quint32 ) ( index / 8 ) ) ] , numBytesPerElement ) ;
tmpbitfield = ( tmpbitfield > > ( index % 8 ) ) & 1 ;
return QVariant ( tmpbitfield ) ;
break ;
}
2012-02-05 14:41:23 +00:00
case STRING :
{
data [ offset + numElements - 1 ] = ' \0 ' ;
QString str ( ( char * ) & data [ offset ] ) ;
return QVariant ( str ) ;
break ;
}
}
// If this point is reached then we got an invalid type
return QVariant ( ) ;
}
bool UAVObjectField : : checkValue ( const QVariant & value , quint32 index )
{
QMutexLocker locker ( obj - > getMutex ( ) ) ;
// Check that index is not out of bounds
if ( index > = numElements )
{
return false ;
}
// Get metadata
UAVObject : : Metadata mdata = obj - > getMetadata ( ) ;
// Update value if the access mode permits
2012-02-20 18:45:18 -07:00
if ( UAVObject : : GetFlightAccess ( mdata ) = = UAVObject : : ACCESS_READWRITE )
2012-02-05 14:41:23 +00:00
{
switch ( type )
{
2010-06-23 02:33:53 +00:00
case INT8 :
case INT16 :
case INT32 :
case UINT8 :
case UINT16 :
case UINT32 :
case FLOAT32 :
case STRING :
2012-05-15 18:22:36 +02:00
case BITFIELD :
2010-06-23 02:33:53 +00:00
return true ;
break ;
2012-02-05 14:41:23 +00:00
case ENUM :
{
qint8 tmpenum = options . indexOf ( value . toString ( ) ) ;
return ( ( tmpenum < 0 ) ? false : true ) ;
break ;
}
2010-06-23 02:33:53 +00:00
default :
2012-02-05 14:41:23 +00:00
qDebug ( ) < < " checkValue: other types " < < type ;
Q_ASSERT ( 0 ) ; // To catch any programming errors where we tried to test invalid values
break ;
}
2010-06-23 02:33:53 +00:00
}
2012-02-05 14:41:23 +00:00
return true ;
2010-06-23 02:33:53 +00:00
}
2012-02-05 14:41:23 +00:00
void UAVObjectField : : setValue ( const QVariant & value , quint32 index )
2010-06-23 02:33:53 +00:00
{
QMutexLocker locker ( obj - > getMutex ( ) ) ;
// Check that index is not out of bounds
if ( index > = numElements )
{
2012-02-05 14:41:23 +00:00
return ;
2010-06-23 02:33:53 +00:00
}
2012-02-05 14:41:23 +00:00
// Get metadata
UAVObject : : Metadata mdata = obj - > getMetadata ( ) ;
// Update value if the access mode permits
2012-02-20 18:45:18 -07:00
if ( UAVObject : : GetGcsAccess ( mdata ) = = UAVObject : : ACCESS_READWRITE )
2010-06-23 02:33:53 +00:00
{
2012-02-05 14:41:23 +00:00
switch ( type )
{
2010-06-23 02:33:53 +00:00
case INT8 :
{
2012-02-05 14:41:23 +00:00
qint8 tmpint8 = value . toInt ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpint8 , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case INT16 :
{
2012-02-05 14:41:23 +00:00
qint16 tmpint16 = value . toInt ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpint16 , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case INT32 :
{
2012-02-05 14:41:23 +00:00
qint32 tmpint32 = value . toInt ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpint32 , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case UINT8 :
{
2012-02-05 14:41:23 +00:00
quint8 tmpuint8 = value . toUInt ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpuint8 , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case UINT16 :
{
2012-02-05 14:41:23 +00:00
quint16 tmpuint16 = value . toUInt ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpuint16 , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case UINT32 :
{
2012-02-05 14:41:23 +00:00
quint32 tmpuint32 = value . toUInt ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpuint32 , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case FLOAT32 :
{
2012-02-05 14:41:23 +00:00
float tmpfloat = value . toFloat ( ) ;
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpfloat , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
case ENUM :
{
2012-02-05 14:41:23 +00:00
qint8 tmpenum = options . indexOf ( value . toString ( ) ) ;
Q_ASSERT ( tmpenum > = 0 ) ; // To catch any programming errors where we set invalid values
memcpy ( & data [ offset + numBytesPerElement * index ] , & tmpenum , numBytesPerElement ) ;
2010-06-23 02:33:53 +00:00
break ;
}
2012-05-15 18:22:36 +02:00
case BITFIELD :
{
quint8 tmpbitfield ;
memcpy ( & tmpbitfield , & data [ offset + numBytesPerElement * ( ( quint32 ) ( index / 8 ) ) ] , numBytesPerElement ) ;
2012-05-15 19:01:32 +02:00
tmpbitfield = ( tmpbitfield & ~ ( 1 < < ( index % 8 ) ) ) | ( ( value . toUInt ( ) ! = 0 ? 1 : 0 ) < < ( index % 8 ) ) ;
2012-05-15 18:22:36 +02:00
memcpy ( & data [ offset + numBytesPerElement * ( ( quint32 ) ( index / 8 ) ) ] , & tmpbitfield , numBytesPerElement ) ;
break ;
}
2010-06-23 02:33:53 +00:00
case STRING :
{
2012-02-05 14:41:23 +00:00
QString str = value . toString ( ) ;
QByteArray barray = str . toAscii ( ) ;
quint32 index ;
for ( index = 0 ; index < ( quint32 ) barray . length ( ) & & index < ( numElements - 1 ) ; + + index )
2011-12-06 22:20:46 -07:00
{
2012-02-05 14:41:23 +00:00
data [ offset + index ] = barray [ index ] ;
2011-12-06 22:20:46 -07:00
}
2012-02-05 14:41:23 +00:00
barray [ index ] = ' \0 ' ;
break ;
2011-12-06 22:20:46 -07:00
}
2010-06-23 02:33:53 +00:00
}
}
}
double UAVObjectField : : getDouble ( quint32 index )
{
2010-08-31 05:09:57 +00:00
return getValue ( index ) . toDouble ( ) ;
2010-06-23 02:33:53 +00:00
}
void UAVObjectField : : setDouble ( double value , quint32 index )
{
2010-08-28 18:48:51 +00:00
setValue ( QVariant ( value ) , index ) ;
2010-06-23 02:33:53 +00:00
}