LCOV - code coverage report
Current view: top level - src/utils/emissions - EnergyParams.cpp (source / functions) Coverage Total Hit
Test: lcov.info Lines: 97.4 % 78 76
Test Date: 2025-11-13 15:38:19 Functions: 100.0 % 12 12

            Line data    Source code
       1              : /****************************************************************************/
       2              : // Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
       3              : // Copyright (C) 2001-2025 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    EnergyParams.cpp
      15              : /// @author  Jakob Erdmann
      16              : /// @author  Michael Behrisch
      17              : /// @date    Sept 2021
      18              : ///
      19              : // A class for parameters used by the emission models
      20              : /****************************************************************************/
      21              : #include <config.h>
      22              : 
      23              : #include <utils/common/MsgHandler.h>
      24              : #include <utils/common/StringUtils.h>
      25              : #include <utils/common/ToString.h>
      26              : #include <utils/geom/GeomHelper.h>
      27              : #include <utils/vehicle/SUMOVTypeParameter.h>
      28              : 
      29              : #include "PollutantsInterface.h"
      30              : #include "HelpersEnergy.h"
      31              : #include "EnergyParams.h"
      32              : 
      33              : 
      34              : // ===========================================================================
      35              : // static definitions
      36              : // ===========================================================================
      37              : const EnergyParams* EnergyParams::myDefault = nullptr;
      38              : const std::vector<SumoXMLAttr> EnergyParams::myParamAttrs = {
      39              :     SUMO_ATTR_SHUT_OFF_STOP, SUMO_ATTR_SHUT_OFF_AUTO,
      40              :     SUMO_ATTR_LOADING, SUMO_ATTR_FRONTSURFACEAREA, SUMO_ATTR_AIRDRAGCOEFFICIENT,
      41              :     SUMO_ATTR_CONSTANTPOWERINTAKE, SUMO_ATTR_WHEELRADIUS, SUMO_ATTR_ROLLDRAGCOEFFICIENT, SUMO_ATTR_ROTATINGMASS,
      42              :     SUMO_ATTR_RADIALDRAGCOEFFICIENT, SUMO_ATTR_PROPULSIONEFFICIENCY, SUMO_ATTR_RECUPERATIONEFFICIENCY,
      43              :     SUMO_ATTR_RECUPERATIONEFFICIENCY_BY_DECELERATION,
      44              :     SUMO_ATTR_MAXIMUMTORQUE, SUMO_ATTR_MAXIMUMPOWER, SUMO_ATTR_GEAREFFICIENCY, SUMO_ATTR_GEARRATIO,
      45              :     SUMO_ATTR_MAXIMUMRECUPERATIONTORQUE, SUMO_ATTR_MAXIMUMRECUPERATIONPOWER,
      46              :     SUMO_ATTR_INTERNALBATTERYRESISTANCE, SUMO_ATTR_NOMINALBATTERYVOLTAGE, SUMO_ATTR_INTERNALMOMENTOFINERTIA
      47              : };
      48              : 
      49              : 
      50              : // ===========================================================================
      51              : // method definitions
      52              : // ===========================================================================
      53       299130 : EnergyParams::EnergyParams(const SUMOVTypeParameter* typeParams) {
      54       598260 :     myCharacteristicMapMap.insert(std::pair<SumoXMLAttr, CharacteristicMap>(SUMO_ATTR_POWERLOSSMAP, CharacteristicMap("2,1|-1e9,1e9;-1e9,1e9|0,0,0,0")));  // P_loss_EM = 0 W for all operating points in the default EV power loss map
      55              : 
      56       299130 :     if (typeParams == nullptr) {
      57         1957 :         myMap[SUMO_ATTR_MASS] = DEFAULT_VEH_MASS;
      58         1957 :         myHaveDefaultMass = true;
      59         1957 :         myMap[SUMO_ATTR_FRONTSURFACEAREA] = DEFAULT_VEH_WIDTH * DEFAULT_VEH_HEIGHT * M_PI / 4.;
      60         1957 :         myHaveDefaultFrontSurfaceArea = true;
      61              :     } else {
      62      6834979 :         for (SumoXMLAttr attr : myParamAttrs) {
      63     13075612 :             if (typeParams->hasParameter(toString(attr))) {
      64         2325 :                 myMap[attr] = typeParams->getDouble(toString(attr), INVALID_DOUBLE);
      65              :             }
      66              :         }
      67       594346 :         for (auto item : myCharacteristicMapMap) {
      68       594346 :             std::string characteristicMapString = typeParams->getParameter(toString(item.first), "");
      69       297173 :             if (characteristicMapString != "") {
      70           60 :                 myCharacteristicMapMap.at(item.first) = CharacteristicMap(typeParams->getParameter(toString(item.first)));
      71              :             }
      72              :         }
      73       297173 :         myMap[SUMO_ATTR_MASS] = typeParams->mass;
      74       297173 :         myHaveDefaultMass = !typeParams->wasSet(VTYPEPARS_MASS_SET);
      75       297173 :         if (myHaveDefaultMass) {
      76       296848 :             const double ecMass = PollutantsInterface::getWeight(typeParams->emissionClass);
      77       296848 :             if (ecMass != -1.) {
      78          137 :                 myMap[SUMO_ATTR_MASS] = ecMass;
      79              :             }
      80              :         }
      81              :         if (myMap.count(SUMO_ATTR_FRONTSURFACEAREA) == 0) {
      82       296919 :             myHaveDefaultFrontSurfaceArea = true;
      83       296919 :             myMap[SUMO_ATTR_FRONTSURFACEAREA] = typeParams->width * typeParams->height * M_PI / 4.;
      84              :         }
      85       297173 :         const std::string& ecName = PollutantsInterface::getName(typeParams->emissionClass);
      86       297173 :         if (typeParams->vehicleClass != SVC_IGNORING && (typeParams->vehicleClass & (SVC_PRIVATE | SVC_VIP | SVC_PASSENGER | SVC_HOV | SVC_TAXI | SVC_E_VEHICLE | SVC_CUSTOM1 | SVC_CUSTOM2)) == 0 && myHaveDefaultFrontSurfaceArea) {
      87       379286 :             if (StringUtils::startsWith(ecName, "MMPEVEM") || StringUtils::startsWith(ecName, "Energy")) {
      88           51 :                 WRITE_WARNINGF(TL("Vehicle type '%' uses the emission class '%' which does not have proper defaults for its vehicle class. "
      89              :                                   "Please use a different emission model or complete the vType definition with further parameters."), typeParams->id, ecName);
      90           17 :                 if (!typeParams->wasSet(VTYPEPARS_MASS_SET)) {
      91           10 :                     WRITE_WARNING(TL(" And also set a vehicle mass!"));
      92              :                 }
      93              :             }
      94              :         }
      95       594346 :         if (!StringUtils::startsWith(ecName, "MMPEVEM")) {
      96              :             if (myMap.count(SUMO_ATTR_INTERNALMOMENTOFINERTIA) > 0) {
      97            6 :                 WRITE_WARNINGF(TL("Vehicle type '%' uses the Energy model parameter 'internalMomentOfInertia' which is deprecated. Use 'rotatingMass' instead."), typeParams->id);
      98            4 :                 if (!typeParams->hasParameter(toString(SUMO_ATTR_ROTATINGMASS))) {
      99            2 :                     myMap[SUMO_ATTR_ROTATINGMASS] = myMap[SUMO_ATTR_INTERNALMOMENTOFINERTIA];
     100              :                 }
     101              :             }
     102              :         }
     103              :     }
     104       299130 : }
     105              : 
     106              : 
     107      1210178 : EnergyParams::~EnergyParams() {}
     108              : 
     109              : 
     110              : void
     111    129075290 : EnergyParams::setDynamicValues(const SUMOTime stopDuration, const bool parking, const SUMOTime waitingTime, const double angle) {
     112    129075290 :     if ((stopDuration >= 0 && myStopDurationSeconds < 0.) || (stopDuration < 0 && myStopDurationSeconds >= 0.)) {
     113        15787 :         myStopDurationSeconds = STEPS2TIME(stopDuration);
     114        15787 :         myAmParking = parking;
     115              :     }
     116    129075290 :     myWaitingTimeSeconds = STEPS2TIME(waitingTime);
     117    129075290 :     myLastAngle = myAngle;
     118    129075290 :     myAngle = angle;
     119    129075290 : }
     120              : 
     121              : 
     122              : void
     123          498 : EnergyParams::setMass(const double mass) {
     124          498 :     myMap[SUMO_ATTR_MASS] = mass;
     125          498 :     myHaveDefaultMass = false;
     126          498 : }
     127              : 
     128              : 
     129              : void
     130         7467 : EnergyParams::setTransportableMass(const double mass) {
     131         7467 :     myTransportableMass = mass;
     132              :     assert(myTransportableMass >= 0);
     133         7467 : }
     134              : 
     135              : 
     136              : double
     137       626497 : EnergyParams::getTotalMass(const double defaultEmptyMass, const double defaultLoading) const {
     138       626497 :     return getDoubleOptional(SUMO_ATTR_MASS, defaultEmptyMass) + getDoubleOptional(SUMO_ATTR_LOADING, defaultLoading) + getTransportableMass();
     139              : }
     140              : 
     141              : 
     142              : double
     143       621147 : EnergyParams::getAngleDiff() const {
     144       621147 :     return myLastAngle == INVALID_DOUBLE ? 0. : GeomHelper::angleDiff(myLastAngle, myAngle);
     145              : }
     146              : 
     147              : 
     148              : double
     149   1514406674 : EnergyParams::getDouble(SumoXMLAttr attr) const {
     150              :     auto it = myMap.find(attr);
     151   1514406674 :     if (it != myMap.end()) {
     152    766870222 :         return it->second;
     153              :     }
     154    747536452 :     if (mySecondaryParams != nullptr) {
     155    747536452 :         return mySecondaryParams->getDouble(attr);
     156              :     }
     157            0 :     throw UnknownElement("Unknown emission model parameter: " + toString(attr));
     158              : }
     159              : 
     160              : 
     161              : double
     162   2912505951 : EnergyParams::getDoubleOptional(SumoXMLAttr attr, const double def) const {
     163              :     auto it = myMap.find(attr);
     164   2912505951 :     if (it != myMap.end() && it->second != INVALID_DOUBLE) {
     165      2431071 :         if (attr == SUMO_ATTR_MASS) {
     166       976333 :             if (!myHaveDefaultMass) {
     167              :                 return it->second;
     168              :             }
     169      1454738 :         } else if (attr == SUMO_ATTR_FRONTSURFACEAREA) {
     170       915878 :             if (!myHaveDefaultFrontSurfaceArea) {
     171              :                 return it->second;
     172              :             }
     173              :         } else {
     174              :             return it->second;
     175              :         }
     176              :     }
     177   2911785406 :     if (mySecondaryParams != nullptr) {
     178   1446870210 :         return mySecondaryParams->getDoubleOptional(attr, def);
     179              :     }
     180              :     return def;
     181              : }
     182              : 
     183              : 
     184              : const CharacteristicMap&
     185        10560 : EnergyParams::getCharacteristicMap(SumoXMLAttr attr) const {
     186              :     auto it = myCharacteristicMapMap.find(attr);
     187        10560 :     if (it != myCharacteristicMapMap.end()) {
     188         5280 :         return it->second;
     189              :     }
     190         5280 :     if (mySecondaryParams != nullptr) {
     191         5280 :         return mySecondaryParams->getCharacteristicMap(attr);
     192              :     }
     193            0 :     throw UnknownElement("Unknown emission model parameter: " + toString(attr));
     194              : }
     195              : 
     196              : 
     197              : bool
     198    737738818 : EnergyParams::isEngineOff() const {
     199   1457510933 :     return myStopDurationSeconds > getDoubleOptional(SUMO_ATTR_SHUT_OFF_STOP, DEFAULT_VEH_SHUT_OFF_STOP) ||
     200    719772115 :            myWaitingTimeSeconds > getDoubleOptional(SUMO_ATTR_SHUT_OFF_AUTO, std::numeric_limits<double>::max());
     201              : }
     202              : 
     203              : 
     204              : bool
     205       698342 : EnergyParams::isOff() const {
     206       698342 :     return myStopDurationSeconds > getDoubleOptional(SUMO_ATTR_SHUT_OFF_STOP, DEFAULT_VEH_SHUT_OFF_STOP);
     207              : }
     208              : 
     209              : 
     210              : /****************************************************************************/
        

Generated by: LCOV version 2.0-1