LCOV - code coverage report
Current view: top level - src/tools - emissionsDrivingCycle_main.cpp (source / functions) Coverage Total Hit
Test: lcov.info Lines: 84.1 % 176 148
Test Date: 2024-12-21 15:45:41 Functions: 100.0 % 1 1

            Line data    Source code
       1              : /****************************************************************************/
       2              : // Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
       3              : // Copyright (C) 2013-2024 German Aerospace Center (DLR) and others.
       4              : // This program and the accompanying materials are made available under the
       5              : // terms of the Eclipse Public License 2.0 which is available at
       6              : // https://www.eclipse.org/legal/epl-2.0/
       7              : // This Source Code may also be made available under the following Secondary
       8              : // Licenses when the conditions for such availability set forth in the Eclipse
       9              : // Public License 2.0 are satisfied: GNU General Public License, version 2
      10              : // or later which is available at
      11              : // https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html
      12              : // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later
      13              : /****************************************************************************/
      14              : /// @file    emissionsDrivingCycle_main.cpp
      15              : /// @author  Daniel Krajzewicz
      16              : /// @author  Michael Behrisch
      17              : /// @date    Wed, 21.08.2013
      18              : ///
      19              : // Main for an emissions calculator
      20              : /****************************************************************************/
      21              : #include <config.h>
      22              : 
      23              : #ifdef HAVE_VERSION_H
      24              : #include <version.h>
      25              : #endif
      26              : 
      27              : #include <iostream>
      28              : #include <string>
      29              : #include <ctime>
      30              : #include <memory>
      31              : #include <utils/common/MsgHandler.h>
      32              : #include <utils/options/Option.h>
      33              : #include <utils/options/OptionsCont.h>
      34              : #include <utils/options/OptionsIO.h>
      35              : #include <utils/common/UtilExceptions.h>
      36              : #include <utils/common/SystemFrame.h>
      37              : #include <utils/common/ToString.h>
      38              : #include <utils/xml/XMLSubSys.h>
      39              : #include <utils/common/FileHelpers.h>
      40              : #include <utils/common/StringTokenizer.h>
      41              : #include <utils/common/StringUtils.h>
      42              : #include <utils/emissions/PollutantsInterface.h>
      43              : #include <utils/emissions/EnergyParams.h>
      44              : #include <utils/iodevices/OutputDevice.h>
      45              : #include <utils/importio/LineReader.h>
      46              : #include "TrajectoriesHandler.h"
      47              : #include "VTypesHandler.h"
      48              : 
      49              : 
      50              : // ===========================================================================
      51              : // functions
      52              : // ===========================================================================
      53              : 
      54              : 
      55              : /* -------------------------------------------------------------------------
      56              :  * main
      57              :  * ----------------------------------------------------------------------- */
      58              : int
      59          996 : main(int argc, char** argv) {
      60          996 :     OptionsCont& oc = OptionsCont::getOptions();
      61          996 :     oc.setApplicationDescription(TL("Computes emissions by driving a time line using SUMO's emission models."));
      62         1992 :     oc.setApplicationName("emissionsDrivingCycle", "Eclipse SUMO emissionsDrivingCycle Version " VERSION_STRING);
      63              : 
      64              :     // add options
      65          996 :     SystemFrame::addConfigurationOptions(oc);
      66          996 :     oc.addOptionSubTopic("Input");
      67          996 :     oc.doRegister("timeline-file", 't', new Option_FileName());
      68         1992 :     oc.addSynonyme("timeline", "timeline-file");
      69         1992 :     oc.addDescription("timeline-file", "Input", TL("Defines the file to read the driving cycle from."));
      70              : 
      71          996 :     oc.doRegister("timeline-file.skip", new Option_Integer(0));
      72         1992 :     oc.addSynonyme("timeline.skip", "timeline-file.skip");
      73         1992 :     oc.addDescription("timeline-file.skip", "Input", TL("Skips the first NUM lines."));
      74              : 
      75         1992 :     oc.doRegister("timeline-file.separator", new Option_String(";"));
      76         1992 :     oc.addSynonyme("timeline.separator", "timeline-file.separator");
      77         1992 :     oc.addDescription("timeline-file.separator", "Input", TL("Defines the entry separator."));
      78              : 
      79          996 :     oc.doRegister("netstate-file", 'n', new Option_FileName());
      80         1992 :     oc.addSynonyme("netstate", "netstate-file");
      81         1992 :     oc.addSynonyme("amitran", "netstate-file");
      82         1992 :     oc.addDescription("netstate-file", "Input", TL("Defines the netstate, route and trajectory files to read the driving cycles from."));
      83              : 
      84          996 :     oc.doRegister("additional-files", new Option_FileName());
      85         1992 :     oc.addDescription("additional-files", "Input", TL("Load emission parameters (vTypes) from FILE(s)"));
      86              : 
      87         1992 :     oc.doRegister("emission-class", 'e', new Option_String("unknown"));
      88         1992 :     oc.addDescription("emission-class", "Input", TL("Defines for which emission class the emissions shall be generated. "));
      89              : 
      90          996 :     oc.doRegister("vtype", new Option_String());
      91         1992 :     oc.addDescription("vtype", "Input", TL("Defines the vehicle type to use for emission parameters."));
      92              : 
      93          996 :     oc.addOptionSubTopic("Processing");
      94          996 :     oc.doRegister("compute-a", 'a', new Option_Bool(false));
      95         1992 :     oc.addDescription("compute-a", "Processing", TL("If set, the acceleration is computed instead of being read from the file. "));
      96              : 
      97          996 :     oc.doRegister("compute-a.forward", new Option_Bool(false));
      98         1992 :     oc.addDescription("compute-a.forward", "Processing", TL("If set, the acceleration for time t is computed from v(t+1) - v(t) instead of v(t) - v(t-1). "));
      99              : 
     100          996 :     oc.doRegister("compute-a.zero-correction", new Option_Bool(false));
     101         1992 :     oc.addDescription("compute-a.zero-correction", "Processing", TL("If set, the acceleration for time t is set to 0 if the speed is 0. "));
     102              : 
     103          996 :     oc.doRegister("skip-first", 's', new Option_Bool(false));
     104         1992 :     oc.addDescription("skip-first", "Processing", TL("If set, the first line of the read file is skipped."));
     105              : 
     106          996 :     oc.doRegister("kmh", new Option_Bool(false));
     107         1992 :     oc.addDescription("kmh", "Processing", TL("If set, the given speed is interpreted as being given in km/h."));
     108              : 
     109          996 :     oc.doRegister("have-slope", new Option_Bool(false));
     110         1992 :     oc.addDescription("have-slope", "Processing", TL("If set, the fourth column is read and used as slope (in deg)."));
     111              : 
     112          996 :     oc.doRegister("slope", new Option_Float(0));
     113         1992 :     oc.addDescription("slope", "Processing", TL("Sets a global slope (in deg) that is used if the file does not contain slope information."));
     114              : 
     115          996 :     oc.addOptionSubTopic("Output");
     116          996 :     oc.doRegister("output-file", 'o', new Option_String());
     117         1992 :     oc.addSynonyme("output", "output-file");
     118         1992 :     oc.addDescription("output", "Output", TL("Defines the file to write the emission cycle results into."));
     119              : 
     120          996 :     oc.doRegister("output.attributes", new Option_StringVector());
     121         1992 :     oc.addDescription("output.attributes", "Output", TL("Defines the attributes to write."));
     122              : 
     123          996 :     oc.doRegister("emission-output", new Option_FileName());
     124         1992 :     oc.addDescription("emission-output", "Output", TL("Save the emission values of each vehicle in XML"));
     125              : 
     126          996 :     oc.doRegister("sum-output", new Option_FileName());
     127         1992 :     oc.addSynonyme("sum", "sum-output");
     128         1992 :     oc.addDescription("sum-output", "Output", TL("Save the aggregated and normed emission values of each vehicle in CSV"));
     129              : 
     130          996 :     oc.addOptionSubTopic("Emissions");
     131          996 :     oc.doRegister("emissions.volumetric-fuel", new Option_Bool(false));
     132         1992 :     oc.addDescription("emissions.volumetric-fuel", "Emissions", TL("Return fuel consumption values in (legacy) unit l instead of mg"));
     133              : 
     134         3984 :     oc.doRegister("phemlight-path", new Option_FileName(StringVector({ "./PHEMlight/" })));
     135         1992 :     oc.addDescription("phemlight-path", "Emissions", TL("Determines where to load PHEMlight definitions from"));
     136              : 
     137          996 :     oc.doRegister("phemlight-year", new Option_Integer(0));
     138         1992 :     oc.addDescription("phemlight-year", "Emissions", TL("Enable fleet age modelling with the given reference year in PHEMlight5"));
     139              : 
     140          996 :     oc.doRegister("phemlight-temperature", new Option_Float(INVALID_DOUBLE));
     141         1992 :     oc.addDescription("phemlight-temperature", "Emissions", TL("Set ambient temperature to correct NOx emissions in PHEMlight5"));
     142              : 
     143         1992 :     oc.doRegister("begin", new Option_String("0", "TIME"));
     144         1992 :     oc.addDescription("begin", "Processing", TL("Defines the begin time in seconds;"));
     145              : 
     146         1992 :     oc.doRegister("end", new Option_String("-1", "TIME"));
     147         1992 :     oc.addDescription("end", "Processing", TL("Defines the end time in seconds;"));
     148              : 
     149          996 :     SystemFrame::addReportOptions(oc);
     150          996 :     oc.doRegister("quiet", 'q', new Option_Bool(false));
     151         1992 :     oc.addDescription("quiet", "Report", TL("Not writing anything."));
     152              : 
     153              :     // run
     154              :     int ret = 0;
     155              :     bool quiet = false;
     156              :     try {
     157              :         // initialise the application system (messaging, xml, options)
     158          996 :         XMLSubSys::init();
     159          996 :         OptionsIO::setArgs(argc, argv);
     160          996 :         OptionsIO::getOptions();
     161          996 :         if (oc.processMetaOptions(argc < 2)) {
     162            0 :             SystemFrame::close();
     163            0 :             return 0;
     164              :         }
     165              : 
     166          996 :         quiet = oc.getBool("quiet");
     167         1000 :         if (!oc.isSet("timeline-file") && !oc.isSet("netstate-file")) {
     168            0 :             throw ProcessError(TL("Either a timeline or a netstate / amitran file must be given."));
     169              :         }
     170         1000 :         if (!oc.isSet("output-file") && (oc.isSet("timeline-file") || !oc.isSet("emission-output"))) {
     171            0 :             throw ProcessError(TL("The output file must be given."));
     172              :         }
     173              :         std::ostream* out = nullptr;
     174         1992 :         if (oc.isSet("output-file")) {
     175         1988 :             out = new std::ofstream(oc.getString("output-file").c_str());
     176              :         }
     177              :         long long int attributes = 0;
     178         1992 :         if (oc.isSet("output.attributes")) {
     179            9 :             for (std::string attrName : oc.getStringVector("output.attributes")) {
     180            3 :                 if (!SUMOXMLDefinitions::Attrs.hasString(attrName)) {
     181            0 :                     if (attrName == "all") {
     182              :                         attributes = std::numeric_limits<long long int>::max() - 1;
     183              :                     } else {
     184            0 :                         WRITE_ERRORF(TL("Unknown attribute '%' to write in output."), attrName);
     185              :                     }
     186              :                     continue;
     187              :                 }
     188            3 :                 int attr = SUMOXMLDefinitions::Attrs.get(attrName);
     189              :                 assert(attr < 63);
     190            3 :                 attributes |= ((long long int)1 << attr);
     191              :             }
     192              :         } else {
     193              :             attributes = ~(((long long int)1 << SUMO_ATTR_AMOUNT));
     194              :         }
     195         1992 :         OutputDevice::createDeviceByOption("emission-output", "emission-export", "emission_file.xsd");
     196              :         OutputDevice* xmlOut = nullptr;
     197         1992 :         if (oc.isSet("emission-output")) {
     198            4 :             xmlOut = &OutputDevice::getDeviceByOption("emission-output");
     199          994 :         } else if (out == nullptr) {
     200              :             out = &std::cout;
     201              :         }
     202              :         std::ostream* sumOut = nullptr;
     203         1992 :         if (oc.isSet("sum-output")) {
     204            0 :             sumOut = new std::ofstream(oc.getString("sum-output").c_str());
     205              :             (*sumOut) << "Vehicle,Cycle,Time,Speed,Gradient,Acceleration,FC,FCel,CO2,NOx,CO,HC,PM" << std::endl;
     206              :         }
     207              : 
     208          997 :         SUMOEmissionClass emissionClass = PollutantsInterface::getClassByName(oc.getString("emission-class"));
     209          996 :         std::unique_ptr<EnergyParams> energyParams;
     210              :         std::map<std::string, SUMOVTypeParameter*> vTypes;
     211         1990 :         if (oc.isSet("vtype") || oc.isSet("additional-files")) {
     212            4 :             if (!oc.isSet("additional-files")) {
     213            0 :                 throw ProcessError(TL("Option --vtype requires option --additional-files for loading vehicle types"));
     214              :             }
     215            4 :             if (!oc.isUsableFileList("additional-files")) {
     216            0 :                 throw ProcessError();
     217              :             }
     218            6 :             for (const std::string& file : oc.getStringVector("additional-files")) {
     219            2 :                 VTypesHandler typesHandler(file, vTypes);
     220            2 :                 if (!XMLSubSys::runParser(typesHandler, file)) {
     221            0 :                     throw ProcessError(TLF("Loading of % failed.", file));
     222              :                 }
     223            2 :             }
     224            4 :             if (!oc.isSet("vtype") && vTypes.size() != 1) {
     225            0 :                 throw ProcessError(TL("Vehicle type is not unique."));
     226              :             }
     227            8 :             const auto vTypeIt = oc.isSet("vtype") ? vTypes.find(oc.getString("vtype")) : vTypes.begin();
     228            2 :             if (vTypeIt == vTypes.end()) {
     229            0 :                 throw ProcessError(TLF("Vehicle type '%' is not defined.", oc.getString("vtype")));
     230              :             }
     231            4 :             if (oc.isDefault("emission-class")) {
     232            0 :                 emissionClass = vTypeIt->second->emissionClass;
     233              :             }
     234            4 :             energyParams = std::unique_ptr<EnergyParams>(new EnergyParams(vTypeIt->second));
     235              :         } else {
     236         1988 :             energyParams = std::unique_ptr<EnergyParams>(new EnergyParams());
     237              :         }
     238              : 
     239          997 :         const bool computeA = oc.getBool("compute-a") || oc.getBool("compute-a.forward");
     240         1993 :         TrajectoriesHandler handler(computeA, oc.getBool("compute-a.forward"), oc.getBool("compute-a.zero-correction"), emissionClass, energyParams.get(), attributes, oc.getFloat("slope"), out, xmlOut);
     241              : 
     242         1992 :         if (oc.isSet("timeline-file")) {
     243         1988 :             int skip = oc.getBool("skip-first") ? 1 : oc.getInt("timeline-file.skip");
     244          994 :             const bool inKMH = oc.getBool("kmh");
     245          994 :             const bool haveSlope = oc.getBool("have-slope");
     246              :             double l = 0;
     247              :             double totalA = 0;
     248              :             double totalS = 0;
     249              :             int time = 0;
     250              : 
     251          994 :             LineReader lr(oc.getString("timeline-file"));
     252          994 :             if (!lr.good()) {
     253            2 :                 throw ProcessError(TLF("Unreadable file '%'.", lr.getFileName()));
     254              :             }
     255      1172733 :             while (lr.hasMore()) {
     256      1171740 :                 std::string line = lr.readLine();
     257      1171740 :                 if (skip > 0) {
     258            0 :                     skip--;
     259              :                     continue;
     260              :                 }
     261      2343480 :                 StringTokenizer st(StringUtils::prune(line), oc.getString("timeline-file.separator"));
     262      1171740 :                 if (st.hasNext()) {
     263              :                     try {
     264      1171740 :                         double t = StringUtils::toDouble(st.next());
     265      1171740 :                         double v = 0;
     266      1171740 :                         if (st.hasNext()) {
     267      1171740 :                             v = StringUtils::toDouble(st.next());
     268              :                         } else {
     269            0 :                             v = t;
     270            0 :                             t = time;
     271              :                         }
     272      1171740 :                         if (inKMH) {
     273      1171740 :                             v /= 3.6;
     274              :                         }
     275      2343480 :                         double a = !computeA && st.hasNext() ? StringUtils::toDouble(st.next()) : TrajectoriesHandler::INVALID_VALUE;
     276      2343480 :                         double s = haveSlope && st.hasNext() ? StringUtils::toDouble(st.next()) : TrajectoriesHandler::INVALID_VALUE;
     277      2343480 :                         if (handler.writeEmissions(*out, "", emissionClass, energyParams.get(), attributes, t, v, a, s)) {
     278      1170747 :                             l += v;
     279      1170747 :                             totalA += a;
     280      1170747 :                             totalS += s;
     281      1170747 :                             time++;
     282              :                         }
     283            0 :                     } catch (EmptyData&) {
     284            0 :                         throw ProcessError(TLF("Missing an entry in line '%'.", line));
     285            0 :                     } catch (NumberFormatException&) {
     286            0 :                         throw ProcessError(TLF("Not numeric entry in line '%'.", line));
     287            0 :                     }
     288              :                 }
     289      1171740 :             }
     290          993 :             if (!quiet) {
     291              :                 std::cout << "sums" << std::endl
     292              :                           << "length:" << l << std::endl;
     293              :             }
     294          993 :             if (sumOut != nullptr) {
     295            0 :                 (*sumOut) << oc.getString("emission-class") << "," << lr.getFileName() << "," << time << ","
     296            0 :                           << (l / time * 3.6) << "," << (totalS / time) << "," << (totalA / time) << ",";
     297            0 :                 handler.writeNormedSums(*sumOut, "", l);
     298              :             }
     299          994 :         }
     300         1990 :         if (oc.isSet("netstate-file")) {
     301            4 :             XMLSubSys::runParser(handler, oc.getString("netstate-file"));
     302              :         }
     303          995 :         if (!quiet) {
     304         1990 :             handler.writeSums(std::cout, "");
     305              :         }
     306          995 :         delete sumOut;
     307          995 :         if (out != &std::cout) {
     308          995 :             delete out;
     309              :         }
     310         1993 :     } catch (InvalidArgument& e) {
     311            0 :         MsgHandler::getErrorInstance()->inform(e.what());
     312            0 :         MsgHandler::getErrorInstance()->inform("Quitting (on error).", false);
     313              :         ret = 1;
     314            1 :     } catch (ProcessError& e) {
     315            3 :         if (std::string(e.what()) != std::string("Process Error") && std::string(e.what()) != std::string("")) {
     316            2 :             MsgHandler::getErrorInstance()->inform(e.what());
     317              :         }
     318            1 :         MsgHandler::getErrorInstance()->inform("Quitting (on error).", false);
     319              :         ret = 1;
     320              : #ifndef _DEBUG
     321            1 :     } catch (...) {
     322            0 :         MsgHandler::getErrorInstance()->inform("Quitting (on unknown error).", false);
     323              :         ret = 1;
     324              : #endif
     325            0 :     }
     326          996 :     SystemFrame::close();
     327          996 :     if (ret == 0 && !quiet) {
     328              :         std::cout << "Success." << std::endl;
     329              :     }
     330              :     return ret;
     331         1992 : }
     332              : 
     333              : 
     334              : /****************************************************************************/
        

Generated by: LCOV version 2.0-1