LCOV - code coverage report
Current view: top level - src/microsim/output - MSMeanData.cpp (source / functions) Coverage Total Hit
Test: lcov.info Lines: 95.5 % 376 359
Test Date: 2026-03-26 16:31:35 Functions: 86.5 % 37 32

            Line data    Source code
       1              : /****************************************************************************/
       2              : // Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
       3              : // Copyright (C) 2001-2026 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    MSMeanData.cpp
      15              : /// @author  Daniel Krajzewicz
      16              : /// @author  Jakob Erdmann
      17              : /// @author  Michael Behrisch
      18              : /// @author  Laura Bieker
      19              : /// @author  Leonhard Luecken
      20              : /// @date    Mon, 10.05.2004
      21              : ///
      22              : // Data collector for edges/lanes
      23              : /****************************************************************************/
      24              : #include <config.h>
      25              : 
      26              : #include <limits>
      27              : #ifdef HAVE_FOX
      28              : #include <utils/common/ScopedLocker.h>
      29              : #endif
      30              : #include <utils/common/SUMOTime.h>
      31              : #include <utils/common/ToString.h>
      32              : #include <utils/common/StringTokenizer.h>
      33              : #include <utils/iodevices/OutputDevice.h>
      34              : #include <microsim/MSEdgeControl.h>
      35              : #include <microsim/MSEdge.h>
      36              : #include <microsim/MSLane.h>
      37              : #include <microsim/MSVehicle.h>
      38              : #include <microsim/cfmodels/MSCFModel.h>
      39              : #include <microsim/MSNet.h>
      40              : #include "MSMeanData_Amitran.h"
      41              : #include "MSMeanData.h"
      42              : 
      43              : #include <microsim/MSGlobals.h>
      44              : #include <mesosim/MESegment.h>
      45              : #include <mesosim/MELoop.h>
      46              : 
      47              : 
      48              : // ===========================================================================
      49              : // debug constants
      50              : // ===========================================================================
      51              : //#define DEBUG_NOTIFY_MOVE
      52              : //#define DEBUG_NOTIFY_ENTER
      53              : 
      54              : // ===========================================================================
      55              : // method definitions
      56              : // ===========================================================================
      57              : // ---------------------------------------------------------------------------
      58              : // MSMeanData::MeanDataValues - methods
      59              : // ---------------------------------------------------------------------------
      60      9134498 : MSMeanData::MeanDataValues::MeanDataValues(
      61              :     MSLane* const lane, const double length, const bool doAdd,
      62      9134498 :     const MSMeanData* const parent) :
      63     18268996 :     MSMoveReminder("meandata_" + (parent == nullptr ? "" : parent->getID() + "|") + (lane == nullptr ? "NULL" :  lane->getID()), lane, doAdd),
      64      9134498 :     myParent(parent),
      65      9134498 :     myLaneLength(length),
      66      9134498 :     sampleSeconds(0),
      67     27403494 :     travelledDistance(0) { }
      68              : 
      69              : 
      70      9133344 : MSMeanData::MeanDataValues::~MeanDataValues() {
      71      9133344 : }
      72              : 
      73              : 
      74              : bool
      75       692908 : MSMeanData::MeanDataValues::notifyEnter(SUMOTrafficObject& veh, MSMoveReminder::Notification reason, const MSLane* enteredLane) {
      76              : #ifdef DEBUG_NOTIFY_ENTER
      77              :     std::cout << "\n" << SIMTIME << " MSMeanData_Net::MSLaneMeanDataValues: veh '" << veh.getID() << "' enters lane '" << enteredLane->getID() << "'" << std::endl;
      78              : #else
      79              :     UNUSED_PARAMETER(enteredLane);
      80              : #endif
      81              :     UNUSED_PARAMETER(reason);
      82       692908 :     return myParent == nullptr || myParent->vehicleApplies(veh);
      83              : }
      84              : 
      85              : 
      86              : bool
      87    485803559 : MSMeanData::MeanDataValues::notifyMove(SUMOTrafficObject& veh, double oldPos, double newPos, double newSpeed) {
      88              :     // if the vehicle has arrived, the reminder must be kept so it can be
      89              :     // notified of the arrival subsequently
      90    485803559 :     const double oldSpeed = veh.getPreviousSpeed();
      91    485803559 :     double enterSpeed = MSGlobals::gSemiImplicitEulerUpdate ? newSpeed : oldSpeed; // NOTE: For the euler update, the vehicle is assumed to travel at constant speed for the whole time step
      92              :     double leaveSpeed = newSpeed, leaveSpeedFront = newSpeed;
      93              : 
      94              :     // These values will be further decreased below
      95    485803559 :     double timeOnLane = TS;
      96    485803559 :     double frontOnLane = oldPos > myLaneLength ? 0. : TS;
      97              :     bool ret = true;
      98              : 
      99              :     // entry and exit times (will be modified below)
     100              :     double timeBeforeEnter = 0.;
     101              :     double timeBeforeEnterBack = 0.;
     102    485803559 :     double timeBeforeLeaveFront = newPos <= myLaneLength ? TS : 0.;
     103              :     double timeBeforeLeave = TS;
     104              : 
     105              :     // Treat the case that the vehicle entered the lane in the last step
     106    485803559 :     if (oldPos < 0 && newPos >= 0) {
     107              :         // Vehicle was not on this lane in the last time step
     108     12222549 :         timeBeforeEnter = MSCFModel::passingTime(oldPos, 0, newPos, oldSpeed, newSpeed);
     109     12222549 :         timeOnLane = TS - timeBeforeEnter;
     110              :         frontOnLane = timeOnLane;
     111     12222549 :         enterSpeed = MSCFModel::speedAfterTime(timeBeforeEnter, oldSpeed, newPos - oldPos);
     112              :     }
     113              : 
     114    485803559 :     const double oldBackPos = oldPos - veh.getVehicleType().getLength();
     115    485803559 :     const double newBackPos = newPos - veh.getVehicleType().getLength();
     116              : 
     117              :     // Determine the time before the vehicle back enters
     118    485803559 :     if (oldBackPos < 0. && newBackPos > 0.) {
     119     12516704 :         timeBeforeEnterBack = MSCFModel::passingTime(oldBackPos, 0., newBackPos, oldSpeed, newSpeed);
     120    473286855 :     } else if (newBackPos <= 0) {
     121      8907874 :         timeBeforeEnterBack = TS;
     122              :     } else {
     123              :         timeBeforeEnterBack = 0.;
     124              :     }
     125              : 
     126              :     // Treat the case that the vehicle's back left the lane in the last step
     127    485803559 :     if (newBackPos > myLaneLength // vehicle's back has left the lane
     128     14251233 :             && oldBackPos <= myLaneLength) { // and hasn't left the lane before
     129              :         assert(!MSGlobals::gSemiImplicitEulerUpdate || newSpeed != 0); // how could it move across the lane boundary otherwise
     130              :         // (Leo) vehicle left this lane (it can also have skipped over it in one time step -> therefore we use "timeOnLane -= ..." and ( ... - timeOnLane) below)
     131     14251233 :         timeBeforeLeave = MSCFModel::passingTime(oldBackPos, myLaneLength, newBackPos, oldSpeed, newSpeed);
     132     14251233 :         const double timeAfterLeave = TS - timeBeforeLeave;
     133     14251233 :         timeOnLane -= timeAfterLeave;
     134     14251233 :         leaveSpeed = MSCFModel::speedAfterTime(timeBeforeLeave, oldSpeed, newPos - oldPos);
     135              :         // XXX: Do we really need this? Why would this "reduce rounding errors"? (Leo) Refs. #2579
     136     14251233 :         if (fabs(timeOnLane) < NUMERICAL_EPS) { // reduce rounding errors
     137              :             timeOnLane = 0.;
     138              :         }
     139     14251233 :         ret = veh.hasArrived();
     140              :     }
     141              : 
     142              :     // Treat the case that the vehicle's front left the lane in the last step
     143    485803559 :     if (newPos > myLaneLength && oldPos <= myLaneLength) {
     144              :         // vehicle's front has left the lane and has not left before
     145              :         assert(!MSGlobals::gSemiImplicitEulerUpdate || newSpeed != 0);
     146     15670023 :         timeBeforeLeaveFront = MSCFModel::passingTime(oldPos, myLaneLength, newPos, oldSpeed, newSpeed);
     147     15670023 :         const double timeAfterLeave = TS - timeBeforeLeaveFront;
     148     15670023 :         frontOnLane -= timeAfterLeave;
     149              :         // XXX: Do we really need this? Why would this "reduce rounding errors"? (Leo) Refs. #2579
     150     15670023 :         if (fabs(frontOnLane) < NUMERICAL_EPS) { // reduce rounding errors
     151              :             frontOnLane = 0.;
     152              :         }
     153     15670023 :         leaveSpeedFront = MSCFModel::speedAfterTime(timeBeforeLeaveFront, oldSpeed, newPos - oldPos);
     154              :     }
     155              : 
     156              :     assert(frontOnLane <= TS);
     157              :     assert(timeOnLane <= TS);
     158              : 
     159    485803559 :     if (timeOnLane < 0) {
     160            0 :         WRITE_ERRORF(TL("Negative vehicle step fraction for '%' on lane '%'."), veh.getID(), getLane()->getID());
     161            0 :         return veh.hasArrived();
     162              :     }
     163    485803559 :     if (timeOnLane == 0) {
     164        22567 :         return veh.hasArrived();
     165              :     }
     166              : 
     167              : #ifdef DEBUG_NOTIFY_MOVE
     168              :     std::stringstream ss;
     169              :     ss << "\n"
     170              :        << "lane length: " << myLaneLength
     171              :        << "\noldPos: " << oldPos
     172              :        << "\nnewPos: " << newPos
     173              :        << "\noldPosBack: " << oldBackPos
     174              :        << "\nnewPosBack: " << newBackPos
     175              :        << "\ntimeBeforeEnter: " << timeBeforeEnter
     176              :        << "\ntimeBeforeEnterBack: " << timeBeforeEnterBack
     177              :        << "\ntimeBeforeLeaveFront: " << timeBeforeLeaveFront
     178              :        << "\ntimeBeforeLeave: " << timeBeforeLeave;
     179              :     if (!(timeBeforeLeave >= MAX2(timeBeforeEnterBack, timeBeforeLeaveFront))
     180              :             || !(timeBeforeEnter <= MIN2(timeBeforeEnterBack, timeBeforeLeaveFront))) {
     181              :         WRITE_ERROR(ss.str());
     182              :     } else {
     183              :         std::cout << ss.str() << std::endl;
     184              :     }
     185              : 
     186              : #endif
     187              : 
     188              :     assert(timeBeforeEnter <= MIN2(timeBeforeEnterBack, timeBeforeLeaveFront));
     189              :     assert(timeBeforeLeave >= MAX2(timeBeforeEnterBack, timeBeforeLeaveFront));
     190              :     // compute average vehicle length on lane in last step
     191    485780992 :     double vehLength = veh.getVehicleType().getLength();
     192              :     // occupied lane length at timeBeforeEnter (resp. stepStart if already on lane)
     193    485780992 :     double lengthOnLaneAtStepStart = MAX2(0., MIN4(myLaneLength, vehLength, vehLength - (oldPos - myLaneLength), oldPos));
     194              :     // occupied lane length at timeBeforeLeave (resp. stepEnd if still on lane)
     195    485780992 :     double lengthOnLaneAtStepEnd = MAX2(0., MIN4(myLaneLength, vehLength, vehLength - (newPos - myLaneLength), newPos));
     196              :     double integratedLengthOnLane = 0.;
     197    485780992 :     if (timeBeforeEnterBack < timeBeforeLeaveFront) {
     198              :         // => timeBeforeLeaveFront>0, myLaneLength>vehLength
     199              :         // vehicle length on detector at timeBeforeEnterBack
     200    468579402 :         double lengthOnLaneAtBackEnter = MIN2(veh.getVehicleType().getLength(), newPos);
     201              :         // linear quadrature of occupancy between timeBeforeEnter and timeBeforeEnterBack
     202    468579402 :         integratedLengthOnLane += (timeBeforeEnterBack - timeBeforeEnter) * (lengthOnLaneAtBackEnter + lengthOnLaneAtStepStart) * 0.5;
     203              :         // linear quadrature of occupancy between timeBeforeEnterBack and timeBeforeLeaveFront
     204              :         // (vehicle is completely on the edge in between)
     205    468579402 :         integratedLengthOnLane += (timeBeforeLeaveFront - timeBeforeEnterBack) * vehLength;
     206              :         // and until vehicle leaves/stepEnd
     207    468579402 :         integratedLengthOnLane += (timeBeforeLeave - timeBeforeLeaveFront) * (vehLength + lengthOnLaneAtStepEnd) * 0.5;
     208     17201590 :     } else if (timeBeforeEnterBack >= timeBeforeLeaveFront) {
     209              :         // => myLaneLength <= vehLength or (timeBeforeLeaveFront == timeBeforeEnterBack == 0)
     210              :         // vehicle length on detector at timeBeforeLeaveFront
     211              :         double lengthOnLaneAtLeaveFront;
     212     17201590 :         if (timeBeforeLeaveFront == timeBeforeEnter) {
     213              :             // for the case that front already left
     214              :             lengthOnLaneAtLeaveFront = lengthOnLaneAtStepStart;
     215      8931926 :         } else if (timeBeforeLeaveFront == timeBeforeLeave) {
     216              :             // for the case that front doesn't leave in this step
     217              :             lengthOnLaneAtLeaveFront = lengthOnLaneAtStepEnd;
     218              :         } else {
     219              :             lengthOnLaneAtLeaveFront = myLaneLength;
     220              :         }
     221              : #ifdef DEBUG_NOTIFY_MOVE
     222              :         std::cout << "lengthOnLaneAtLeaveFront=" << lengthOnLaneAtLeaveFront << std::endl;
     223              : #endif
     224              :         // linear quadrature of occupancy between timeBeforeEnter and timeBeforeLeaveFront
     225     17201590 :         integratedLengthOnLane += (timeBeforeLeaveFront - timeBeforeEnter) * (lengthOnLaneAtLeaveFront + lengthOnLaneAtStepStart) * 0.5;
     226              :         // linear quadrature of occupancy between timeBeforeLeaveFront and timeBeforeEnterBack
     227     17201590 :         integratedLengthOnLane += (timeBeforeEnterBack - timeBeforeLeaveFront) * lengthOnLaneAtLeaveFront;
     228              :         // and until vehicle leaves/stepEnd
     229     17201590 :         integratedLengthOnLane += (timeBeforeLeave - timeBeforeEnterBack) * (lengthOnLaneAtLeaveFront + lengthOnLaneAtStepEnd) * 0.5;
     230              :     }
     231              : 
     232    485780992 :     double meanLengthOnLane = integratedLengthOnLane / TS;
     233              : #ifdef DEBUG_NOTIFY_MOVE
     234              :     std::cout << "Calculated mean length on lane '" << myLane->getID() << "' in last step as " << meanLengthOnLane
     235              :               << "\nlengthOnLaneAtStepStart=" << lengthOnLaneAtStepStart << ", lengthOnLaneAtStepEnd=" << lengthOnLaneAtStepEnd << ", integratedLengthOnLane=" << integratedLengthOnLane
     236              :               << std::endl;
     237              : #endif
     238              : 
     239              : //    // XXX: use this, when #2556 is fixed! Refs. #2575
     240              : //    const double travelledDistanceFrontOnLane = MAX2(0., MIN2(newPos, myLaneLength) - MAX2(oldPos, 0.));
     241              : //    const double travelledDistanceVehicleOnLane = MIN2(newPos, myLaneLength) - MAX2(oldPos, 0.) + MIN2(MAX2(0., newPos - myLaneLength), veh.getVehicleType().getLength());
     242              : //    // XXX: #2556 fixed for ballistic update
     243    485780992 :     const double travelledDistanceFrontOnLane = MSGlobals::gSemiImplicitEulerUpdate ? frontOnLane * newSpeed
     244    161836490 :             : MAX2(0., MIN2(newPos, myLaneLength) - MAX2(oldPos, 0.));
     245    566699237 :     const double travelledDistanceVehicleOnLane = MSGlobals::gSemiImplicitEulerUpdate ? timeOnLane * newSpeed
     246    161836490 :             : MIN2(newPos, myLaneLength) - MAX2(oldPos, 0.) + MIN2(MAX2(0., newPos - myLaneLength), veh.getVehicleType().getLength());
     247              : //    // XXX: no fix
     248              : //    const double travelledDistanceFrontOnLane = frontOnLane*newSpeed;
     249              : //    const double travelledDistanceVehicleOnLane = timeOnLane*newSpeed;
     250              : 
     251              : #ifdef HAVE_FOX
     252    485780992 :     ScopedLocker<> lock(myNotificationMutex, MSGlobals::gNumSimThreads > 1);
     253              : #endif
     254    485780992 :     notifyMoveInternal(veh, frontOnLane, timeOnLane, (enterSpeed + leaveSpeedFront) / 2., (enterSpeed + leaveSpeed) / 2., travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
     255              :     return ret;
     256              : }
     257              : 
     258              : 
     259              : bool
     260      1183717 : MSMeanData::MeanDataValues::notifyLeave(SUMOTrafficObject& /*veh*/, double /*lastPos*/, MSMoveReminder::Notification reason, const MSLane* /* enteredLane */) {
     261      1183717 :     if (MSGlobals::gUseMesoSim) {
     262              :         return false; // reminder is re-added on every segment (@recheck for performance)
     263              :     }
     264      1029189 :     return reason == MSMoveReminder::NOTIFICATION_JUNCTION;
     265              : }
     266              : 
     267              : 
     268              : bool
     269      2950375 : MSMeanData::MeanDataValues::isEmpty() const {
     270      2950375 :     return sampleSeconds == 0;
     271              : }
     272              : 
     273              : 
     274              : void
     275            0 : MSMeanData::MeanDataValues::update() {
     276            0 : }
     277              : 
     278              : 
     279              : double
     280      1023760 : MSMeanData::MeanDataValues::getSamples() const {
     281      1023760 :     return sampleSeconds;
     282              : }
     283              : 
     284              : 
     285              : // ---------------------------------------------------------------------------
     286              : // MSMeanData::MeanDataValueTracker - methods
     287              : // ---------------------------------------------------------------------------
     288         2744 : MSMeanData::MeanDataValueTracker::MeanDataValueTracker(MSLane* const lane,
     289              :         const double length,
     290         2744 :         const MSMeanData* const parent)
     291         2744 :     : MSMeanData::MeanDataValues(lane, length, true, parent) {
     292         2744 :     myCurrentData.push_back(std::make_shared<TrackerEntry>(parent->createValues(lane, length, false)));
     293         2744 : }
     294              : 
     295              : 
     296         8232 : MSMeanData::MeanDataValueTracker::~MeanDataValueTracker() {}
     297              : 
     298              : 
     299              : void
     300       164304 : MSMeanData::MeanDataValueTracker::reset(bool afterWrite) {
     301       164304 :     if (afterWrite) {
     302        70360 :         if (!myCurrentData.empty()) {
     303        70360 :             myCurrentData.pop_front();
     304              :         }
     305              :     } else {
     306       187888 :         myCurrentData.push_back(std::make_shared<TrackerEntry>(myParent->createValues(myLane, myLaneLength, false)));
     307              :     }
     308       164304 : }
     309              : 
     310              : 
     311              : void
     312            0 : MSMeanData::MeanDataValueTracker::addTo(MSMeanData::MeanDataValues& val) const {
     313            0 :     myCurrentData.front()->myValues->addTo(val);
     314            0 : }
     315              : 
     316              : 
     317              : void
     318        15022 : MSMeanData::MeanDataValueTracker::notifyMoveInternal(const SUMOTrafficObject& veh, const double frontOnLane, const double timeOnLane, const double meanSpeedFrontOnLane, const double meanSpeedVehicleOnLane, const double travelledDistanceFrontOnLane, const double travelledDistanceVehicleOnLane, const double meanLengthOnLane) {
     319        15022 :     myTrackedData[&veh]->myValues->notifyMoveInternal(veh, frontOnLane, timeOnLane, meanSpeedFrontOnLane, meanSpeedVehicleOnLane, travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
     320        15022 : }
     321              : 
     322              : 
     323              : bool
     324          784 : MSMeanData::MeanDataValueTracker::notifyLeave(SUMOTrafficObject& veh, double lastPos, MSMoveReminder::Notification reason, const MSLane* /* enteredLane */) {
     325          784 :     if (myParent == nullptr || reason != MSMoveReminder::NOTIFICATION_SEGMENT) {
     326          384 :         myTrackedData[&veh]->myNumVehicleLeft++;
     327              :     }
     328          784 :     return myTrackedData[&veh]->myValues->notifyLeave(veh, lastPos, reason);
     329              : }
     330              : 
     331              : 
     332              : bool
     333         3400 : MSMeanData::MeanDataValueTracker::notifyEnter(SUMOTrafficObject& veh, MSMoveReminder::Notification reason, const MSLane* enteredLane) {
     334              : #ifdef DEBUG_NOTIFY_ENTER
     335              :     std::cout << "\n" << SIMTIME << " MSMeanData::MeanDataValueTracker: veh '" << veh.getID() << "' enters lane '" << enteredLane->getID() << "'" << std::endl;
     336              : #else
     337              :     UNUSED_PARAMETER(enteredLane);
     338              : #endif
     339         3400 :     if (reason == MSMoveReminder::NOTIFICATION_SEGMENT) {
     340              :         return true;
     341              :     }
     342         5768 :     if (myParent->vehicleApplies(veh) && myTrackedData.find(&veh) == myTrackedData.end()) {
     343          504 :         myTrackedData[&veh] = myCurrentData.back();
     344          504 :         myTrackedData[&veh]->myNumVehicleEntered++;
     345          504 :         if (!myTrackedData[&veh]->myValues->notifyEnter(veh, reason)) {
     346            0 :             myTrackedData[&veh]->myNumVehicleLeft++;
     347            0 :             myTrackedData.erase(&veh);
     348            0 :             return false;
     349              :         }
     350              :         return true;
     351              :     }
     352              :     return false;
     353              : }
     354              : 
     355              : 
     356              : bool
     357        91696 : MSMeanData::MeanDataValueTracker::isEmpty() const {
     358        91696 :     return myCurrentData.front()->myValues->isEmpty();
     359              : }
     360              : 
     361              : 
     362              : void
     363          368 : MSMeanData::MeanDataValueTracker::write(OutputDevice& dev,
     364              :                                         const SumoXMLAttrMask& attributeMask,
     365              :                                         const SUMOTime period,
     366              :                                         const int numLanes,
     367              :                                         const double speedLimit,
     368              :                                         const double defaultTravelTime,
     369              :                                         const int /*numVehicles*/) const {
     370          368 :     myCurrentData.front()->myValues->write(dev, attributeMask, period, numLanes, speedLimit,
     371              :                                            defaultTravelTime,
     372              :                                            myCurrentData.front()->myNumVehicleEntered);
     373          368 : }
     374              : 
     375              : 
     376              : int
     377        63116 : MSMeanData::MeanDataValueTracker::getNumReady() const {
     378              :     int result = 0;
     379      1445056 :     for (const auto& it : myCurrentData) {
     380      1384592 :         if (it->myNumVehicleEntered == it->myNumVehicleLeft) {
     381      1381940 :             result++;
     382              :         } else {
     383              :             break;
     384              :         }
     385              :     }
     386        63116 :     return result;
     387              : }
     388              : 
     389              : 
     390              : double
     391          368 : MSMeanData::MeanDataValueTracker::getSamples() const {
     392          368 :     return myCurrentData.front()->myValues->getSamples();
     393              : }
     394              : 
     395              : 
     396              : // ---------------------------------------------------------------------------
     397              : // MSMeanData - methods
     398              : // ---------------------------------------------------------------------------
     399        24193 : MSMeanData::MSMeanData(const std::string& id,
     400              :                        const SUMOTime dumpBegin, const SUMOTime dumpEnd,
     401              :                        const bool useLanes, const std::string& excludeEmpty, const bool withInternal,
     402              :                        const bool trackVehicles,
     403              :                        const int detectPersons,
     404              :                        const double maxTravelTime,
     405              :                        const double minSamples,
     406              :                        const std::string& vTypes,
     407              :                        const std::string& writeAttributes,
     408              :                        const std::vector<MSEdge*>& edges,
     409        24193 :                        AggregateType aggregate) :
     410              :     MSDetectorFileOutput(id, vTypes, "", detectPersons),
     411        24193 :     myMinSamples(minSamples),
     412        24193 :     myMaxTravelTime(maxTravelTime),
     413        24193 :     myAmEdgeBased(!useLanes),
     414        24193 :     myDumpBegin(dumpBegin),
     415        24193 :     myDumpEnd(dumpEnd),
     416        24193 :     myInitTime(SUMOTime_MAX),
     417        24193 :     myEdges(edges),
     418          156 :     myDumpInternal(withInternal && MSGlobals::gUsingInternalLanes),
     419        24193 :     myTrackVehicles(trackVehicles),
     420        96772 :     myWrittenAttributes(OutputDevice::parseWrittenAttributes(StringTokenizer(writeAttributes).getVector(), "meandata '" + id + "'")),
     421        72579 :     myAggregate(aggregate) {
     422              :     try {
     423        24193 :         myDumpEmpty = !StringUtils::toBool(excludeEmpty);
     424           84 :     } catch (const BoolFormatException&) {
     425           84 :         if (excludeEmpty == "default" || excludeEmpty == "defaults") {
     426           78 :             myPrintDefaults = true;
     427            6 :         } else if (excludeEmpty == "modified") {
     428            6 :             myDumpEmpty = false;
     429            6 :             myPrintDefaults = true;
     430            6 :             myPrintModified = true;
     431              :         } else {
     432            0 :             throw;
     433              :         }
     434           84 :     }
     435        24193 : }
     436              : 
     437              : 
     438              : void
     439        23270 : MSMeanData::init() {
     440        23270 :     myInitTime = MSNet::getInstance()->getCurrentTimeStep();
     441        23270 :     if (myEdges.empty()) {
     442              :         // use all edges by default
     443       818555 :         for (MSEdge* const edge : MSNet::getInstance()->getEdgeControl().getEdges()) {
     444       795354 :             if ((myDumpInternal || !edge->isInternal()) &&
     445       528404 :                     ((detectsPersons() && myDumpInternal) || (!edge->isCrossing() && !edge->isWalkingArea()))) {
     446       350372 :                 myEdges.push_back(edge);
     447              :             }
     448              :         }
     449              :     }
     450              :     int index = 0;
     451       373755 :     for (MSEdge* edge : myEdges) {
     452       350485 :         myMeasures.push_back(std::vector<MeanDataValues*>());
     453       350485 :         myEdgeIndex[edge] = index++;
     454       350485 :         const std::vector<MSLane*>& lanes = edge->getLanes();
     455       350485 :         if (MSGlobals::gUseMesoSim) {
     456              :             MeanDataValues* data;
     457        29423 :             if (!myAmEdgeBased) {
     458           85 :                 for (MSLane* const lane : lanes) {
     459           52 :                     data = createValues(lane, lanes[0]->getLength(), false);
     460           52 :                     myMeasures.back().push_back(data);
     461           52 :                     MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     462          310 :                     while (s != nullptr) {
     463          258 :                         s->addDetector(data, lane->getIndex());
     464          258 :                         s->prepareDetectorForWriting(*data, lane->getIndex());
     465              :                         s = s->getNextSegment();
     466              :                     }
     467           52 :                     data->reset();
     468           52 :                     data->reset(true);
     469              :                 }
     470              :             } else {
     471        29390 :                 if (myTrackVehicles) {
     472          880 :                     data = new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this);
     473              :                 } else {
     474        28510 :                     data = createValues(nullptr, lanes[0]->getLength(), false);
     475              :                 }
     476        58780 :                 data->setDescription("meandata_" + getID() + "|" + edge->getID());
     477        29390 :                 myMeasures.back().push_back(data);
     478        29390 :                 MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     479       150584 :                 while (s != nullptr) {
     480       121194 :                     s->addDetector(data);
     481       121194 :                     s->prepareDetectorForWriting(*data);
     482              :                     s = s->getNextSegment();
     483              :                 }
     484        29390 :                 data->reset();
     485        29390 :                 data->reset(true);
     486              :             }
     487              :             continue;
     488        29423 :         }
     489       321062 :         if (myAmEdgeBased && myTrackVehicles) {
     490          936 :             myMeasures.back().push_back(new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this));
     491              :         }
     492       848691 :         for (MSLane* const lane : lanes) {
     493       527629 :             if (myTrackVehicles) {
     494         2016 :                 if (myAmEdgeBased) {
     495         1088 :                     lane->addMoveReminder(myMeasures.back().back());
     496              :                 } else {
     497          928 :                     myMeasures.back().push_back(new MeanDataValueTracker(lane, lane->getLength(), this));
     498              :                 }
     499              :             } else {
     500       525613 :                 myMeasures.back().push_back(createValues(lane, lane->getLength(), true));
     501              :             }
     502              :         }
     503              :     }
     504        23270 :     if (myAggregate == AggregateType::TAZ) {
     505          450 :         for (const MSEdge* e : MSEdge::getAllEdges()) {
     506          444 :             if (e->isTazConnector()) {
     507          180 :                 myTAZ.push_back(e);
     508              :             }
     509              :         }
     510              :     }
     511        23270 : }
     512              : 
     513              : 
     514        24138 : MSMeanData::~MSMeanData() {
     515       374014 :     for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
     516       905642 :         for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
     517       555766 :             delete *j;
     518              :         }
     519              :     }
     520        24138 : }
     521              : 
     522              : 
     523              : void
     524        12649 : MSMeanData::resetOnly(SUMOTime stopTime) {
     525              :     UNUSED_PARAMETER(stopTime);
     526        12649 :     if (MSGlobals::gUseMesoSim) {
     527              :         MSEdgeVector::iterator edge = myEdges.begin();
     528       117300 :         for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i, ++edge) {
     529       113716 :             MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(**edge);
     530       227432 :             for (MeanDataValues* data : *i) {
     531       602292 :                 while (s != nullptr) {
     532       488576 :                     s->prepareDetectorForWriting(*data);
     533              :                     s = s->getNextSegment();
     534              :                 }
     535       113716 :                 data->reset();
     536              :             }
     537              :         }
     538              :         return;
     539              :     }
     540       257356 :     for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
     541       530134 :         for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
     542       281843 :             (*j)->reset();
     543              :         }
     544              :     }
     545              : }
     546              : 
     547              : 
     548              : std::string
     549       576253 : MSMeanData::getEdgeID(const MSEdge* const edge) {
     550       576253 :     return edge->getID();
     551              : }
     552              : 
     553              : 
     554              : void
     555          940 : MSMeanData::writeAggregated(OutputDevice& dev, SUMOTime startTime, SUMOTime stopTime) {
     556          940 :     if (myTrackVehicles) {
     557            0 :         throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
     558              :     }
     559              : 
     560              :     double edgeLengthSum = 0;
     561              :     int laneNumber = 0;
     562              :     double speedSum = 0;
     563              :     double totalTT = 0;
     564        42172 :     for (MSEdge* edge : myEdges) {
     565        41232 :         edgeLengthSum += edge->getLength();
     566        41232 :         laneNumber += edge->getNumDrivingLanes();
     567        41232 :         speedSum += edge->getSpeedLimit();
     568        41232 :         totalTT += edge->getLength() / edge->getSpeedLimit();
     569              :     }
     570          940 :     MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
     571        42172 :     for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
     572        82512 :         for (MeanDataValues* meanData : edgeValues) {
     573        41280 :             meanData->addTo(*sumData);
     574        41280 :             if (!MSNet::getInstance()->skipFinalReset()) {
     575        41280 :                 meanData->reset();
     576              :             }
     577              :         }
     578              :     }
     579          940 :     if (MSGlobals::gUseMesoSim) {
     580        14040 :         for (int i = 0; i < (int)myEdges.size(); i++) {
     581        13728 :             MSEdge* edge = myEdges[i];
     582              :             std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
     583        13728 :             MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     584        68640 :             while (s != nullptr) {
     585       109824 :                 for (MeanDataValues* meanData : edgeValues) {
     586        54912 :                     s->prepareDetectorForWriting(*meanData);
     587        54912 :                     meanData->addTo(*sumData);
     588        54912 :                     if (!MSNet::getInstance()->skipFinalReset()) {
     589        54912 :                         meanData->reset();
     590              :                     }
     591              :                 }
     592              :                 s = s->getNextSegment();
     593              :             }
     594              :         }
     595              :     }
     596              : 
     597          940 :     if (myDumpEmpty || !sumData->isEmpty()) {
     598         1868 :         writePrefix(dev, *sumData, SUMO_TAG_EDGE, "AGGREGATED");
     599          934 :         dev.writeAttr(SUMO_ATTR_NUMEDGES, myEdges.size());
     600          934 :         sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)myEdges.size(),
     601          934 :                        myPrintDefaults ? totalTT : -1.);
     602              :     }
     603          940 :     delete sumData;
     604          940 : }
     605              : 
     606              : 
     607              : void
     608           18 : MSMeanData::writeAggregatedTAZ(OutputDevice& dev, SUMOTime startTime, SUMOTime stopTime) {
     609           18 :     if (myTrackVehicles) {
     610            0 :         throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
     611              :     }
     612              : 
     613          558 :     for (const MSEdge* taz : myTAZ) {
     614              :         double edgeLengthSum = 0;
     615              :         int laneNumber = 0;
     616              :         double speedSum = 0;
     617              :         double totalTT = 0;
     618              :         std::set<const MSEdge*> connected;
     619         1332 :         for (const MSEdge* edge : taz->getSuccessors()) {
     620              :             connected.insert(edge);
     621              :         }
     622         1332 :         for (const MSEdge* edge : taz->getPredecessors()) {
     623              :             connected.insert(edge);
     624              :         }
     625         2124 :         for (const MSEdge* edge : connected) {
     626         1584 :             edgeLengthSum += edge->getLength();
     627         1584 :             laneNumber += edge->getNumDrivingLanes();
     628         1584 :             speedSum += edge->getSpeedLimit();
     629         1584 :             totalTT += edge->getLength() / edge->getSpeedLimit();
     630              :         }
     631          540 :         MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
     632        24300 :         for (int i = 0; i < (int)myEdges.size(); i++) {
     633        23760 :             MSEdge* edge = myEdges[i];
     634              :             if (connected.count(edge) != 0) {
     635              :                 std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
     636         1584 :                 if (MSGlobals::gUseMesoSim) {
     637          528 :                     MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     638         2640 :                     while (s != nullptr) {
     639         4224 :                         for (MeanDataValues* meanData : edgeValues) {
     640         2112 :                             s->prepareDetectorForWriting(*meanData);
     641         2112 :                             meanData->addTo(*sumData);
     642              :                         }
     643              :                         s = s->getNextSegment();
     644              :                     }
     645              :                 } else {
     646         2112 :                     for (MeanDataValues* meanData : edgeValues) {
     647         1056 :                         meanData->addTo(*sumData);
     648              :                     }
     649              :                 }
     650              :             }
     651              :         }
     652          540 :         if (myDumpEmpty || !sumData->isEmpty()) {
     653          216 :             writePrefix(dev, *sumData, SUMO_TAG_EDGE, taz->getID());
     654           72 :             dev.writeAttr(SUMO_ATTR_NUMEDGES, connected.size());
     655           72 :             sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)connected.size(),
     656           72 :                            myPrintDefaults ? totalTT : -1.);
     657              :         }
     658          540 :         delete sumData;
     659              :     }
     660              : 
     661           18 :     if (!MSNet::getInstance()->skipFinalReset()) {
     662          810 :         for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
     663         1584 :             for (MeanDataValues* meanData : edgeValues) {
     664          792 :                 meanData->reset();
     665              :             }
     666              :         }
     667              :     }
     668           18 : }
     669              : 
     670              : 
     671              : void
     672     17885744 : MSMeanData::writeEdge(OutputDevice& dev,
     673              :                       const std::vector<MeanDataValues*>& edgeValues,
     674              :                       const MSEdge* const edge, SUMOTime startTime, SUMOTime stopTime) {
     675     17885744 :     if (MSGlobals::gUseMesoSim) {
     676              :         int idx = 0;
     677      2721977 :         for (MeanDataValues* const data : edgeValues) {
     678      1361012 :             MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     679      7002470 :             while (s != nullptr) {
     680      5641800 :                 s->prepareDetectorForWriting(*data, myAmEdgeBased ? -1 : idx);
     681              :                 s = s->getNextSegment();
     682              :             }
     683      1361012 :             idx++;
     684              :         }
     685      1360965 :         if (myAmEdgeBased) {
     686      1360918 :             MeanDataValues* const data = edgeValues.front();
     687      1360918 :             if (myDumpEmpty || (myPrintModified && edge->getLanes()[0]->isSpeedModified()) || !data->isEmpty()) {
     688        60488 :                 writePrefix(dev, *data, SUMO_TAG_EDGE, getEdgeID(edge));
     689        60488 :                 data->write(dev, myWrittenAttributes, stopTime - startTime,
     690              :                             edge->getNumDrivingLanes(),
     691              :                             edge->getSpeedLimit(),
     692        60488 :                             myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
     693              :             }
     694      1360918 :             if (!MSNet::getInstance()->skipFinalReset()) {
     695      1360918 :                 data->reset(true);
     696              :             }
     697      1360918 :             return;
     698              :         }
     699              :     }
     700     16524826 :     if (!myAmEdgeBased) {
     701      8025230 :         bool writeCheck = myDumpEmpty;
     702      8025230 :         if (!writeCheck) {
     703     15611296 :             for (const MeanDataValues* const laneData : edgeValues) {
     704      7912883 :                 if (!laneData->isEmpty() || (myPrintModified && laneData->getLane()->isSpeedModified())) {
     705              :                     writeCheck = true;
     706              :                     break;
     707              :                 }
     708              :             }
     709              :         }
     710      7901535 :         if (writeCheck) {
     711       326817 :             dev.openTag(SUMO_TAG_EDGE).writeAttr(SUMO_ATTR_ID, edge->getID());
     712              :         }
     713     16170242 :         for (MeanDataValues* const laneData : edgeValues) {
     714              :             const MSLane* const lane = laneData->getLane();
     715      8145012 :             if (myDumpEmpty || (myPrintModified && lane->isSpeedModified()) || !laneData->isEmpty()) {
     716       870370 :                 writePrefix(dev, *laneData, SUMO_TAG_LANE, lane->getID());
     717       435185 :                 laneData->write(dev, myWrittenAttributes, stopTime - startTime, 1, lane->getSpeedLimit(),
     718       435185 :                                 myPrintDefaults ? lane->getLength() / lane->getSpeedLimit() : -1.);
     719              :             }
     720      8145012 :             if (!MSNet::getInstance()->skipFinalReset()) {
     721      8145012 :                 laneData->reset(true);
     722              :             }
     723              :         }
     724      8025230 :         if (writeCheck) {
     725       653634 :             dev.closeTag();
     726              :         }
     727              :     } else {
     728      8499596 :         if (myTrackVehicles) {
     729        22440 :             MeanDataValues& meanData = **edgeValues.begin();
     730        22440 :             if (myDumpEmpty || !meanData.isEmpty()) {
     731          160 :                 writePrefix(dev, meanData, SUMO_TAG_EDGE, edge->getID());
     732          160 :                 meanData.write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
     733          160 :                                myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
     734              :             }
     735        22440 :             if (!MSNet::getInstance()->skipFinalReset()) {
     736        22440 :                 meanData.reset(true);
     737              :             }
     738              :         } else {
     739      8477156 :             MeanDataValues* sumData = createValues(nullptr, edge->getLength(), false);
     740      8477156 :             bool writeCheck = myDumpEmpty;
     741     17121836 :             for (MeanDataValues* const laneData : edgeValues) {
     742      8644680 :                 laneData->addTo(*sumData);
     743      8644680 :                 if (myPrintModified && laneData->getLane()->isSpeedModified()) {
     744              :                     writeCheck = true;
     745              :                 }
     746      8644680 :                 if (!MSNet::getInstance()->skipFinalReset()) {
     747      8644680 :                     laneData->reset();
     748              :                 }
     749              :             }
     750      8477156 :             if (writeCheck || !sumData->isEmpty()) {
     751       518497 :                 writePrefix(dev, *sumData, SUMO_TAG_EDGE, getEdgeID(edge));
     752       518497 :                 sumData->write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
     753       518497 :                                myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
     754              :             }
     755      8477155 :             delete sumData;
     756              :         }
     757              :     }
     758              : }
     759              : 
     760              : 
     761              : void
     762       437760 : MSMeanData::openInterval(OutputDevice& dev, const SUMOTime startTime, const SUMOTime stopTime) {
     763       437760 :     dev.openTag(SUMO_TAG_INTERVAL).writeAttr(SUMO_ATTR_BEGIN, time2string(startTime)).writeAttr(SUMO_ATTR_END, time2string(stopTime));
     764       437760 :     dev.writeAttr(SUMO_ATTR_ID, myID);
     765       437760 : }
     766              : 
     767              : 
     768              : void
     769      1012580 : MSMeanData::writePrefix(OutputDevice& dev, const MeanDataValues& values, const SumoXMLTag tag, const std::string id) const {
     770      1012580 :     dev.openTag(tag);
     771      1012580 :     dev.writeAttr(SUMO_ATTR_ID, id);
     772      1012580 :     dev.writeOptionalAttr(SUMO_ATTR_SAMPLEDSECONDS, values.getSamples(), myWrittenAttributes);
     773      1012580 : }
     774              : 
     775              : 
     776              : void
     777       449075 : MSMeanData::writeXMLOutput(OutputDevice& dev,
     778              :                            SUMOTime startTime, SUMOTime stopTime) {
     779              :     // check whether this dump shall be written for the current time
     780       449075 :     int numReady = myDumpBegin < stopTime && myDumpEnd - DELTA_T >= startTime ? 1 : 0;
     781       449075 :     if (myTrackVehicles && myDumpBegin < stopTime) {
     782         2124 :         myPendingIntervals.push_back(std::make_pair(startTime, stopTime));
     783         2124 :         numReady = (int)myPendingIntervals.size();
     784        63312 :         for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
     785       124304 :             for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
     786        63116 :                 numReady = MIN2(numReady, ((MeanDataValueTracker*)*j)->getNumReady());
     787        63116 :                 if (numReady == 0) {
     788              :                     break;
     789              :                 }
     790              :             }
     791        63092 :             if (numReady == 0) {
     792              :                 break;
     793              :             }
     794              :         }
     795              :     }
     796       449075 :     const bool partialInterval = startTime < myInitTime;
     797       449075 :     if (numReady == 0 || myTrackVehicles || partialInterval) {
     798        12649 :         resetOnly(stopTime);
     799              :     }
     800       449075 :     if (partialInterval) {
     801              :         return;
     802              :     }
     803       887078 :     while (numReady-- > 0) {
     804       438014 :         if (!myPendingIntervals.empty()) {
     805         1588 :             startTime = myPendingIntervals.front().first;
     806         1588 :             stopTime = myPendingIntervals.front().second;
     807         1588 :             myPendingIntervals.pop_front();
     808              :         }
     809       438014 :         openInterval(dev, startTime, stopTime);
     810       438014 :         if (myAggregate == AggregateType::YES) {
     811          940 :             writeAggregated(dev, startTime, stopTime);
     812       437074 :         } else if (myAggregate == AggregateType::TAZ) {
     813           18 :             writeAggregatedTAZ(dev, startTime, stopTime);
     814              :         } else {
     815              :             MSEdgeVector::const_iterator edge = myEdges.begin();
     816     18322799 :             for (const std::vector<MeanDataValues*>& measures : myMeasures) {
     817     17885744 :                 writeEdge(dev, measures, *edge, startTime, stopTime);
     818              :                 ++edge;
     819              :             }
     820              :         }
     821       876026 :         dev.closeTag();
     822              :     }
     823              :     dev.flush();
     824              : }
     825              : 
     826              : 
     827              : void
     828        23248 : MSMeanData::writeXMLDetectorProlog(OutputDevice& dev) const {
     829        46496 :     dev.writeXMLHeader("meandata", "meandata_file.xsd");
     830        23248 : }
     831              : 
     832              : 
     833              : void
     834     28296877 : MSMeanData::detectorUpdate(const SUMOTime step) {
     835     28296877 :     if (step + DELTA_T == myDumpBegin) {
     836          639 :         init();
     837              :     }
     838     28296877 : }
     839              : 
     840              : 
     841              : const std::vector<MSMeanData::MeanDataValues*>*
     842            0 : MSMeanData::getEdgeValues(const MSEdge* edge) const {
     843              :     auto it = myEdgeIndex.find(edge);
     844            0 :     if (it != myEdgeIndex.end()) {
     845            0 :         return &myMeasures[it->second];
     846              :     } else {
     847              :         return nullptr;
     848              :     }
     849              : }
     850              : 
     851              : 
     852              : const std::vector<MSMoveReminder*>
     853            1 : MSMeanData::getReminders() const {
     854              :     std::vector<MSMoveReminder*> result;
     855           45 :     for (auto vec : myMeasures) {
     856           44 :         result.insert(result.end(), vec.begin(), vec.end());
     857           44 :     }
     858            1 :     return result;
     859            0 : }
     860              : 
     861              : 
     862              : /****************************************************************************/
        

Generated by: LCOV version 2.0-1