/** ****************************************************************************** * * @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); wiresharkOutputPath.mkpath(wiresharkOutputPath.absolutePath()); wiresharkCodeTemplate = readFile(wiresharkCodePath.absoluteFilePath("op-uavobjects/packet-op-uavobjects.c.template")); 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 = writeFileIfDifferent(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, 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, 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 = writeFileIfDifferent(outputpath.absolutePath() + "/packet-op-uavobjects-" + info->namelc + ".c", outCode); if (!res) { cout << "Error: Could not write wireshark code files" << endl; return false; } return true; }