1
0
mirror of https://bitbucket.org/librepilot/librepilot.git synced 2025-01-18 03:52:11 +01:00
LibrePilot/ground/uavobjgenerator/generators/wireshark/uavobjectgeneratorwireshark.cpp
Stacey Sheldon 4c47a36bda uavobjgenerator: Add generator for wireshark UAVO plugin
Using 'make uavobjects_wireshark' now generates a complete
wireshark dissector that can decode all fields of the available
UAVObjects.
2012-06-19 00:21:35 -04:00

291 lines
12 KiB
C++

/**
******************************************************************************
*
* @file uavobjectgeneratorflight.cpp
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
* @brief produce flight code for uavobjects
*
* @see The GNU Public License (GPL) Version 3
*
*****************************************************************************/
/*
* 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 "uavobjectgeneratorwireshark.h"
using namespace std;
bool UAVObjectGeneratorWireshark::generate(UAVObjectParser* parser,QString templatepath,QString outputpath) {
fieldTypeStrHf << "FT_INT8" << "FT_INT16" << "FT_INT32" <<"FT_UINT8"
<<"FT_UINT16" << "FT_UINT32" << "FT_FLOAT" << "FT_UINT8";
fieldTypeStrGlib << "gint8" << "gint16" << "gint32" <<"guint8"
<<"guint16" << "guint32" << "gfloat" << "guint8";
wiresharkCodePath = QDir( templatepath + QString("ground/openpilotgcs/src/plugins/uavobjects/wireshark"));
wiresharkOutputPath = QDir( outputpath + QString("wireshark") );
wiresharkOutputPath.mkpath(wiresharkOutputPath.absolutePath());
wiresharkCodeTemplate = readFile( wiresharkCodePath.absoluteFilePath("op-uavobjects/packet-op-uavobjects-template.c") );
wiresharkMakeTemplate = readFile( wiresharkCodePath.absoluteFilePath("op-uavobjects/Makefile.common-template") );
if ( wiresharkCodeTemplate.isNull() || wiresharkMakeTemplate.isNull()) {
cerr << "Error: Could not open wireshark template files." << endl;
return false;
}
/* Copy static files for wireshark plugins root directory into output directory */
QStringList topstaticfiles;
topstaticfiles << "Custom.m4" << "Custom.make" << "Custom.nmake";
for (int i = 0; i < topstaticfiles.length(); ++i) {
QFile::copy(wiresharkCodePath.absoluteFilePath(topstaticfiles[i]),
wiresharkOutputPath.absoluteFilePath(topstaticfiles[i]));
}
/* Copy static files for op-uavtalk dissector into output directory */
QDir uavtalkOutputPath = QDir( outputpath + QString("wireshark/op-uavtalk") );
uavtalkOutputPath.mkpath(uavtalkOutputPath.absolutePath());
QStringList uavtalkstaticfiles;
uavtalkstaticfiles << "AUTHORS" << "COPYING" << "ChangeLog";
uavtalkstaticfiles << "CMakeLists.txt" << "Makefile.nmake";
uavtalkstaticfiles << "Makefile.am" << "moduleinfo.h" << "moduleinfo.nmake";
uavtalkstaticfiles << "plugin.rc.in";
uavtalkstaticfiles << "Makefile.common" << "packet-op-uavtalk.c";
for (int i = 0; i < uavtalkstaticfiles.length(); ++i) {
QFile::copy(wiresharkCodePath.absoluteFilePath("op-uavtalk/" + uavtalkstaticfiles[i]),
uavtalkOutputPath.absoluteFilePath(uavtalkstaticfiles[i]));
}
/* Copy static files for op-uavobjects dissector into output directory */
QDir uavobjectsOutputPath = QDir( outputpath + QString("wireshark/op-uavobjects") );
uavobjectsOutputPath.mkpath(uavobjectsOutputPath.absolutePath());
QStringList uavostaticfiles;
uavostaticfiles << "AUTHORS" << "COPYING" << "ChangeLog";
uavostaticfiles << "CMakeLists.txt" << "Makefile.nmake";
uavostaticfiles << "Makefile.am" << "moduleinfo.h" << "moduleinfo.nmake";
uavostaticfiles << "plugin.rc.in";
for (int i = 0; i < uavostaticfiles.length(); ++i) {
QFile::copy(wiresharkCodePath.absoluteFilePath("op-uavobjects/" + uavostaticfiles[i]),
uavobjectsOutputPath.absoluteFilePath(uavostaticfiles[i]));
}
/* Generate the per-object files from the templates, and keep track of the list of generated filenames */
QString objFileNames;
for (int objidx = 0; objidx < parser->getNumObjects(); ++objidx) {
ObjectInfo* info = parser->getObjectByIndex(objidx);
process_object(info, uavobjectsOutputPath);
objFileNames.append(" packet-op-uavobjects-" + info->namelc + ".c");
}
/* Write the uavobject dissector's Makefile.common */
wiresharkMakeTemplate.replace( QString("$(UAVOBJFILENAMES)"), objFileNames);
bool res = writeFileIfDiffrent( uavobjectsOutputPath.absolutePath() + "/Makefile.common",
wiresharkMakeTemplate );
if (!res) {
cout << "Error: Could not write wireshark Makefile" << endl;
return false;
}
return true;
}
/**
* Generate the Flight object files
**/
bool UAVObjectGeneratorWireshark::process_object(ObjectInfo* info, QDir outputpath)
{
if (info == NULL)
return false;
// Prepare output strings
QString outCode = wiresharkCodeTemplate;
// Replace common tags
replaceCommonTags(outCode, info);
// Replace the $(SUBTREES) and $(SUBTREESTATICS) tags
QString subtrees;
QString subtreestatics;
subtreestatics.append( QString("static gint ett_uavo = -1;\r\n") );
subtrees.append( QString("&ett_uavo,\r\n") );
for (int n = 0; n < info->fields.length(); ++n) {
if (info->fields[n]->numElements > 1) {
/* Reserve a subtree for each array */
subtreestatics.append( QString("static gint ett_%1_%2 = -1;\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name) );
subtrees.append( QString("&ett_%1_%2,\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name) );
}
}
outCode.replace(QString("$(SUBTREES)"), subtrees);
outCode.replace(QString("$(SUBTREESTATICS)"), subtreestatics);
// Replace the $(FIELDHANDLES) tag
QString type;
QString fields;
for (int n = 0; n < info->fields.length(); ++n) {
fields.append( QString("static int hf_op_uavobjects_%1_%2 = -1;\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name));
if (info->fields[n]->numElements > 1) {
QStringList elemNames = info->fields[n]->elementNames;
for (int m = 0; m < elemNames.length(); ++m) {
fields.append( QString("static int hf_op_uavobjects_%1_%2_%3 = -1;\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name)
.arg(elemNames[m]) );
}
}
}
outCode.replace(QString("$(FIELDHANDLES)"), fields);
// Replace the $(ENUMFIELDNAMES) tag
QString enums;
for (int n = 0; n < info->fields.length(); ++n) {
// Only for enum types
if (info->fields[n]->type == FIELDTYPE_ENUM) {
enums.append(QString("/* Field %1 information */\r\n").arg(info->fields[n]->name) );
enums.append(QString("/* Enumeration options for field %1 */\r\n").arg(info->fields[n]->name));
enums.append( QString("static const value_string uavobjects_%1_%2[]= {\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name) );
// Go through each option
QStringList options = info->fields[n]->options;
for (int m = 0; m < options.length(); ++m) {
enums.append ( QString("\t{ %1, \"%2\" },\r\n")
.arg(m)
.arg(options[m].replace(QRegExp(ENUM_SPECIAL_CHARS), "") ) );
}
enums.append( QString("\t{ 0, NULL }\r\n") );
enums.append( QString("};\r\n") );
}
}
outCode.replace(QString("$(ENUMFIELDNAMES)"), enums);
// Replace the $(POPULATETREE) tag
QString treefields;
for (int n = 0; n < info->fields.length(); ++n) {
if ( info->fields[n]->numElements == 1 ) {
treefields.append( QString(" ptvcursor_add(cursor, hf_op_uavobjects_%1_%2, sizeof(%3), ENC_LITTLE_ENDIAN);\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name)
.arg(fieldTypeStrGlib[info->fields[n]->type]) );
} else {
treefields.append( QString(" {\r\n") );
treefields.append( QString(" proto_item * it = NULL;\r\n") );
treefields.append( QString(" it = ptvcursor_add_no_advance(cursor, hf_op_uavobjects_%1_%2, 1, ENC_NA);\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name) );
treefields.append( QString(" ptvcursor_push_subtree(cursor, it, ett_%1_%2);\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name) );
/* Populate each array element into the table */
QStringList elemNames = info->fields[n]->elementNames;
for (int m = 0; m < elemNames.length(); ++m) {
treefields.append( QString(" ptvcursor_add(cursor, hf_op_uavobjects_%1_%2_%3, sizeof(%4), ENC_LITTLE_ENDIAN);\r\n")
.arg(info->namelc)
.arg(info->fields[n]->name)
.arg(elemNames[m])
.arg(fieldTypeStrGlib[info->fields[n]->type]) );
}
treefields.append( QString(" ptvcursor_pop_subtree(cursor);\r\n") );
treefields.append( QString(" }\r\n") );
}
}
outCode.replace(QString("$(POPULATETREE)"), treefields);
// Replace the $(HEADERFIELDS) tag
QString headerfields;
headerfields.append( QString(" static hf_register_info hf[] = {\r\n") );
for (int n = 0; n < info->fields.length(); ++n) {
// For non-array fields
if ( info->fields[n]->numElements == 1) {
headerfields.append( QString("\t { &hf_op_uavobjects_%1_%2,\r\n")
.arg( info->namelc )
.arg( info->fields[n]->name ) );
headerfields.append( QString("\t { \"%1\", \"%2.%3\", %4,\r\n")
.arg( info->fields[n]->name )
.arg( info->namelc )
.arg( info->fields[n]->name )
.arg( fieldTypeStrHf[info->fields[n]->type] ) );
if ( info->fields[n]->type == FIELDTYPE_ENUM ) {
headerfields.append( QString("\t BASE_DEC, VALS(uavobjects_%1_%2), 0x0, NULL, HFILL \r\n")
.arg( info->namelc )
.arg( info->fields[n]->name ) );
} else if ( info->fields[n]->type == FIELDTYPE_FLOAT32 ) {
headerfields.append( QString("\t BASE_NONE, NULL, 0x0, NULL, HFILL \r\n") );
} else {
headerfields.append( QString("\t BASE_DEC_HEX, NULL, 0x0, NULL, HFILL\r\n") );
}
headerfields.append( QString("\t },\r\n") );
headerfields.append( QString("\t },\r\n") );
} else {
headerfields.append( QString("\t { &hf_op_uavobjects_%1_%2,\r\n")
.arg( info->namelc )
.arg( info->fields[n]->name ) );
headerfields.append( QString("\t { \"%1\", \"%2.%3\", FT_NONE,\r\n")
.arg( info->fields[n]->name )
.arg( info->namelc )
.arg( info->fields[n]->name ) );
headerfields.append( QString("\t BASE_NONE, NULL, 0x0, NULL, HFILL\r\n") );
headerfields.append( QString("\t },\r\n") );
headerfields.append( QString("\t },\r\n") );
QStringList elemNames = info->fields[n]->elementNames;
for (int m = 0; m < elemNames.length(); ++m) {
headerfields.append( QString("\t { &hf_op_uavobjects_%1_%2_%3,\r\n")
.arg( info->namelc )
.arg( info->fields[n]->name )
.arg( elemNames[m]) );
headerfields.append( QString("\t { \"%1\", \"%2.%3.%4\", %5,\r\n")
.arg( elemNames[m] )
.arg( info->namelc )
.arg( info->fields[n]->name )
.arg( elemNames[m] )
.arg( fieldTypeStrHf[info->fields[n]->type] ) );
if ( info->fields[n]->type == FIELDTYPE_ENUM ) {
headerfields.append( QString("\t BASE_DEC, VALS(uavobjects_%1_%2), 0x0, NULL, HFILL \r\n")
.arg( info->namelc )
.arg( info->fields[n]->name ) );
} else if ( info->fields[n]->type == FIELDTYPE_FLOAT32 ) {
headerfields.append( QString("\t BASE_NONE, NULL, 0x0, NULL, HFILL \r\n") );
} else {
headerfields.append( QString("\t BASE_DEC_HEX, NULL, 0x0, NULL, HFILL\r\n") );
}
headerfields.append( QString("\t },\r\n") );
headerfields.append( QString("\t },\r\n") );
}
}
}
headerfields.append( QString(" };\r\n") );
outCode.replace(QString("$(HEADERFIELDS)"), headerfields);
// Write the flight code
bool res = writeFileIfDiffrent( outputpath.absolutePath() + "/packet-op-uavobjects-" + info->namelc + ".c", outCode );
if (!res) {
cout << "Error: Could not write wireshark code files" << endl;
return false;
}
return true;
}