1
0
mirror of https://bitbucket.org/librepilot/librepilot.git synced 2025-01-25 10:52:11 +01:00

1190 lines
36 KiB
C++

/**
******************************************************************************
*
* @file uavobjectfield.cpp
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
* @see The GNU Public License (GPL) Version 3
* @addtogroup GCSPlugins GCS Plugins
* @{
* @addtogroup UAVObjectsPlugin UAVObjects Plugin
* @{
* @brief The UAVUObjects GCS plugin
*****************************************************************************/
/*
* 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, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "uavobjectfield.h"
#include <QtEndian>
#include <QDebug>
#include <QtWidgets>
UAVObjectField::UAVObjectField(const QString & name, const QString & description, const QString & units, FieldType type, quint32 numElements, const QStringList & options, const QString &limits)
{
QStringList elementNames;
// Set element names
for (quint32 n = 0; n < numElements; ++n) {
elementNames.append(QString("%1").arg(n));
}
// Initialize
constructorInitialize(name, description, units, type, elementNames, options, limits);
}
UAVObjectField::UAVObjectField(const QString & name, const QString & description, const QString & units, FieldType type, const QStringList & elementNames, const QStringList & options, const QString &limits)
{
constructorInitialize(name, description, units, type, elementNames, options, limits);
}
void UAVObjectField::constructorInitialize(const QString & name, const QString & description, const QString & units, FieldType type, const QStringList & elementNames, const QStringList & options, const QString &limits)
{
// Copy params
this->name = name;
this->description = description;
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) {
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;
case BITFIELD:
numBytesPerElement = sizeof(quint8);
this->options = QStringList() << tr("0") << tr("1");
break;
case STRING:
numBytesPerElement = sizeof(quint8);
break;
default:
numBytesPerElement = 0;
}
limitsInitialize(limits);
}
void UAVObjectField::limitsInitialize(const QString &limits)
{
// Limit string format:
// % - start char
// XXXX - optional BOARD_TYPE and BOARD_REVISION (4 hex digits)
// TY - rule type (EQ-equal, NE-not equal, BE-between, BI-bigger, SM-smaller)
// VAL - values for TY separated by colon
// , - rule separator (may have leading or trailing spaces)
// ; - element separator (may have leading or trailing spaces)
//
// Examples:
// Disable few flight modes for Revo (00903):
// "%0903NE:Autotune:VelocityControl:PositionHold:ReturnToBase:Land:PathPlanner"
// Original CC board (rev 1), first element bigger than 3 and second element inside [2.3-5.0]:
// "%0401BI:3; %BE:2.3:5"
// Set applicable range [0-500] for 3 elements of array for all boards:
// "%BE:0:500; %BE:0:500; %BE:0:500"
if (limits.isEmpty()) {
return;
}
QStringList stringPerElement = limits.split(";");
quint32 index = 0;
foreach(QString str, stringPerElement) {
QStringList ruleList = str.split(",");
QList<LimitStruct> limitList;
foreach(QString rule, ruleList) {
QString _str = rule.trimmed();
if (_str.isEmpty()) {
continue;
}
QStringList valuesPerElement = _str.split(":");
LimitStruct lstruc;
bool startFlag = valuesPerElement.at(0).startsWith("%");
bool maxIndexFlag = (int)(index) < (int)numElements;
bool elemNumberSizeFlag = valuesPerElement.at(0).size() == 3;
bool aux;
valuesPerElement.at(0).mid(1, 4).toInt(&aux, 16);
bool b4 = ((valuesPerElement.at(0).size()) == 7 && aux);
if (startFlag && maxIndexFlag && (elemNumberSizeFlag || b4)) {
if (b4) {
lstruc.board = valuesPerElement.at(0).mid(1, 4).toInt(&aux, 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) {
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());
}
}
limitList.append(lstruc);
} else {
if (!valuesPerElement.at(0).isEmpty() && !startFlag) {
qDebug() << "limits parsing failed (property doesn't start with %) on UAVObjectField" << name;
} else if (!maxIndexFlag) {
qDebug() << "limits parsing failed (index>numelements) on UAVObjectField" << name << "index" << index << "numElements" << numElements;
} else if (!elemNumberSizeFlag || !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;
}
}
}
elementLimits.insert(index, limitList);
++index;
}
// foreach(QList<LimitStruct> limitList, elementLimits) {
// foreach(LimitStruct limit, limitList) {
// qDebug() << "Limit type" << limit.type << "for board" << limit.board << "for field" << getName();
// foreach(QVariant var, limit.values) {
// qDebug() << "value" << var;
// }
// }
// }
}
bool UAVObjectField::isWithinLimits(QVariant var, quint32 index, int board)
{
if (!elementLimits.keys().contains(index)) {
return true;
}
foreach(LimitStruct struc, elementLimits.value(index)) {
if ((struc.board != board) && board != 0 && struc.board != 0) {
continue;
}
switch (struc.type) {
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;
}
break;
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;
}
break;
case BETWEEN:
if (struc.values.length() < 2) {
qDebug() << __FUNCTION__ << "between limit with less than 1 pair, aborting; field:" << name;
return true;
}
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;
}
break;
case BIGGER:
if (struc.values.length() < 1) {
qDebug() << __FUNCTION__ << "BIGGER limit with less than 1 value, aborting; field:" << name;
return true;
}
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;
}
break;
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;
}
default:
return true;
}
}
return true;
}
QString UAVObjectField::getLimitsAsString(quint32 index, int board)
{
QString limitString;
if (elementLimits.keys().contains(index)) {
foreach(LimitStruct struc, elementLimits.value(index)) {
if ((struc.board != board) && board != 0 && struc.board != 0) {
continue;
}
switch (struc.type) {
case EQUAL:
{
limitString.append(tr("one of")).append(" [");
bool first = true;
foreach(QVariant var, struc.values) {
if (!first) {
limitString.append(", ");
}
limitString.append(var.toString());
first = false;
}
return limitString.append("]");
}
case NOT_EQUAL:
{
limitString.append(tr("none of")).append(" [");
bool first = true;
foreach(QVariant var, struc.values) {
if (!first) {
limitString.append(", ");
}
limitString.append(var.toString());
first = false;
}
return limitString.append("]");
}
case BIGGER: return limitString.append(QString("%1 %2").arg(tr("more than"), struc.values.at(0).toString()));
case BETWEEN: return limitString.append(QString("%1 %2 %3 %4")
.arg(tr("between"), struc.values.at(0).toString(),
tr(" and "), struc.values.at(1).toString()));
case SMALLER: return limitString.append(QString("%1 %2").arg(tr("less than"), struc.values.at(0).toString()));
default:
break;
}
}
}
return limitString;
}
QVariant UAVObjectField::getMaxLimit(quint32 index, int board)
{
if (!elementLimits.keys().contains(index)) {
return QVariant();
}
foreach(LimitStruct struc, elementLimits.value(index)) {
if ((struc.board != board) && board != 0 && struc.board != 0) {
continue;
}
switch (struc.type) {
case EQUAL:
case NOT_EQUAL:
case BIGGER:
return QVariant();
case BETWEEN:
return struc.values.at(1);
case SMALLER:
return struc.values.at(0);
default:
return QVariant();
}
}
return QVariant();
}
QVariant UAVObjectField::getMinLimit(quint32 index, int board)
{
if (!elementLimits.keys().contains(index)) {
return QVariant();
}
foreach(LimitStruct struc, elementLimits.value(index)) {
if ((struc.board != board) && board != 0 && struc.board != 0) {
return QVariant();
}
switch (struc.type) {
case EQUAL:
case NOT_EQUAL:
case SMALLER:
return QVariant();
case BETWEEN:
return struc.values.at(0);
case BIGGER:
return struc.values.at(0);
default:
return QVariant();
}
}
return QVariant();
}
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;
}
QString UAVObjectField::getTypeAsString()
{
switch (type) {
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";
case UAVObjectField::BITFIELD:
return "bitfield";
case UAVObjectField::STRING:
return "string";
default:
return "";
}
}
QStringList UAVObjectField::getElementNames()
{
return elementNames;
}
UAVObject *UAVObjectField::getObject()
{
return obj;
}
void UAVObjectField::clear()
{
QMutexLocker locker(obj->getMutex());
switch (type) {
case BITFIELD:
memset(&data[offset], 0, numBytesPerElement * ((quint32)(1 + (numElements - 1) / 8)));
break;
default:
memset(&data[offset], 0, numBytesPerElement * numElements);
break;
}
}
QString UAVObjectField::getName()
{
return name;
}
QString UAVObjectField::getDescription()
{
return description;
}
QString UAVObjectField::getUnits()
{
return units;
}
QStringList UAVObjectField::getOptions()
{
return options;
}
quint32 UAVObjectField::getNumElements()
{
return numElements;
}
quint32 UAVObjectField::getDataOffset()
{
return offset;
}
quint32 UAVObjectField::getNumBytes()
{
switch (type) {
case BITFIELD:
return numBytesPerElement * ((quint32)(1 + (numElements - 1) / 8));
break;
default:
return numBytesPerElement * numElements;
break;
}
}
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;
}
void UAVObjectField::toXML(QXmlStreamWriter *xmlWriter)
{
xmlWriter->writeStartElement("field");
xmlWriter->writeAttribute("name", getName());
xmlWriter->writeAttribute("type", getTypeAsString());
if (!getUnits().isEmpty()) {
xmlWriter->writeAttribute("unit", getUnits());
}
for (unsigned int n = 0; n < numElements; ++n) {
xmlWriter->writeStartElement("value");
if (getElementNames().size() > 1) {
xmlWriter->writeAttribute("name", getElementNames().at(n));
}
xmlWriter->writeCharacters(getValue(n).toString());
xmlWriter->writeEndElement(); // value
}
xmlWriter->writeEndElement(); // field
}
void UAVObjectField::fromXML(QXmlStreamReader *xmlReader)
{
// Assert we have the correct field by name
Q_ASSERT(xmlReader->name() == "field");
Q_ASSERT(xmlReader->attributes().value("name") == getName());
// Read values, skip overflowing ones if any
while (xmlReader->readNextStartElement()) {
if (xmlReader->name() == "value") {
int index = getElementNames().indexOf(xmlReader->attributes().value("name").toString());
if (index >= 0) {
setValue(xmlReader->readElementText(), index);
}
}
}
}
void UAVObjectField::toJson(QJsonObject &jsonObject)
{
jsonObject["name"] = getName();
jsonObject["type"] = getTypeAsString();
jsonObject["unit"] = getUnits();
QJsonArray values;
for (unsigned int n = 0; n < numElements; ++n) {
QJsonObject value;
value["name"] = getElementNames().at(n);
value["value"] = QJsonValue::fromVariant(getValue(n));
values.append(value);
}
jsonObject["values"] = values;
}
void UAVObjectField::fromJson(const QJsonObject &jsonObject)
{
Q_ASSERT(jsonObject["name"].toString() == getName());
QJsonArray jsonValues = jsonObject["values"].toArray();
for (int i = 0; i < jsonValues.size(); i++) {
QJsonObject jsonValue = jsonValues.at(i).toObject();
int index = getElementNames().indexOf(jsonValue["name"].toString());
if (index >= 0) {
setValue(((QJsonValue)jsonValue["value"]).toVariant(), index);
}
}
}
qint32 UAVObjectField::pack(quint8 *dataOut)
{
QMutexLocker locker(obj->getMutex());
// Pack each element in output buffer
switch (type) {
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;
case BITFIELD:
for (quint32 index = 0; index < (quint32)(1 + (numElements - 1) / 8); ++index) {
dataOut[numBytesPerElement * index] = data[offset + numBytesPerElement * index];
}
break;
case STRING:
memcpy(dataOut, &data[offset], numElements);
break;
}
// Done
return getNumBytes();
}
qint32 UAVObjectField::unpack(const quint8 *dataIn)
{
QMutexLocker locker(obj->getMutex());
// Unpack each element from input buffer
switch (type) {
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;
case BITFIELD:
for (quint32 index = 0; index < (quint32)(1 + (numElements - 1) / 8); ++index) {
data[offset + numBytesPerElement * index] = dataIn[numBytesPerElement * index];
}
break;
case STRING:
memcpy(&data[offset], dataIn, numElements);
break;
}
// Done
return getNumBytes();
}
bool UAVObjectField::isNumeric()
{
switch (type) {
case INT8:
case INT16:
case INT32:
case UINT8:
case UINT16:
case UINT32:
case FLOAT32:
case BITFIELD:
return true;
break;
default:
return false;
}
}
bool UAVObjectField::isInteger()
{
switch (type) {
case INT8:
case INT16:
case INT32:
case UINT8:
case UINT16:
case UINT32:
return true;
break;
default:
return false;
}
}
bool UAVObjectField::isText()
{
switch (type) {
case ENUM:
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);
if (tmpenum >= options.length()) {
qDebug() << "Invalid value for" << name;
tmpenum = 0;
}
return QVariant(options[tmpenum]);
break;
}
case BITFIELD:
{
quint8 tmpbitfield;
memcpy(&tmpbitfield, &data[offset + numBytesPerElement * ((quint32)(index / 8))], numBytesPerElement);
tmpbitfield = (tmpbitfield >> (index % 8)) & 1;
return QVariant(tmpbitfield);
break;
}
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
if (UAVObject::GetFlightAccess(mdata) == UAVObject::ACCESS_READWRITE) {
switch (type) {
case INT8:
case INT16:
case INT32:
case UINT8:
case UINT16:
case UINT32:
case FLOAT32:
case STRING:
case BITFIELD:
return true;
break;
case ENUM:
{
qint8 tmpenum = options.indexOf(value.toString());
return (tmpenum < 0) ? false : true;
break;
}
default:
qDebug() << "checkValue: other types" << type;
Q_ASSERT(0); // To catch any programming errors where we tried to test invalid values
break;
}
}
return true;
}
void UAVObjectField::setValue(const QVariant & value, quint32 index)
{
QMutexLocker locker(obj->getMutex());
// Check that index is not out of bounds
if (index >= numElements) {
return;
}
// Get metadata
UAVObject::Metadata mdata = obj->getMetadata();
// Update value if the access mode permits
if (UAVObject::GetGcsAccess(mdata) == UAVObject::ACCESS_READWRITE) {
switch (type) {
case INT8:
{
qint8 tmpint8 = value.toInt();
memcpy(&data[offset + numBytesPerElement * index], &tmpint8, numBytesPerElement);
break;
}
case INT16:
{
qint16 tmpint16 = value.toInt();
memcpy(&data[offset + numBytesPerElement * index], &tmpint16, numBytesPerElement);
break;
}
case INT32:
{
qint32 tmpint32 = value.toInt();
memcpy(&data[offset + numBytesPerElement * index], &tmpint32, numBytesPerElement);
break;
}
case UINT8:
{
quint8 tmpuint8 = value.toUInt();
memcpy(&data[offset + numBytesPerElement * index], &tmpuint8, numBytesPerElement);
break;
}
case UINT16:
{
quint16 tmpuint16 = value.toUInt();
memcpy(&data[offset + numBytesPerElement * index], &tmpuint16, numBytesPerElement);
break;
}
case UINT32:
{
quint32 tmpuint32 = value.toUInt();
memcpy(&data[offset + numBytesPerElement * index], &tmpuint32, numBytesPerElement);
break;
}
case FLOAT32:
{
float tmpfloat = value.toFloat();
memcpy(&data[offset + numBytesPerElement * index], &tmpfloat, numBytesPerElement);
break;
}
case ENUM:
{
qint8 tmpenum = options.indexOf(value.toString());
// Default to 0 on invalid values.
if (tmpenum < 0) {
tmpenum = 0;
}
memcpy(&data[offset + numBytesPerElement * index], &tmpenum, numBytesPerElement);
break;
}
case BITFIELD:
{
quint8 tmpbitfield;
memcpy(&tmpbitfield, &data[offset + numBytesPerElement * ((quint32)(index / 8))], numBytesPerElement);
tmpbitfield = (tmpbitfield & ~(1 << (index % 8))) | ((value.toUInt() != 0 ? 1 : 0) << (index % 8));
memcpy(&data[offset + numBytesPerElement * ((quint32)(index / 8))], &tmpbitfield, numBytesPerElement);
break;
}
case STRING:
{
QString str = value.toString();
QByteArray barray = str.toLatin1();
quint32 index;
for (index = 0; index < (quint32)barray.length() && index < (numElements - 1); ++index) {
data[offset + index] = barray[index];
}
barray[index] = '\0';
break;
}
}
}
}
double UAVObjectField::getDouble(quint32 index)
{
return getValue(index).toDouble();
}
void UAVObjectField::setDouble(double value, quint32 index)
{
setValue(QVariant(value), index);
}