LCOV - code coverage report
Current view: top level - src/microsim/output - MSMeanData.cpp (source / functions) Coverage Total Hit
Test: lcov.info Lines: 95.4 % 327 312
Test Date: 2025-11-14 15:59:05 Functions: 86.1 % 36 31

            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    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      9042550 : MSMeanData::MeanDataValues::MeanDataValues(
      61              :     MSLane* const lane, const double length, const bool doAdd,
      62      9042550 :     const MSMeanData* const parent) :
      63     18085100 :     MSMoveReminder("meandata_" + (parent == nullptr ? "" : parent->getID() + "|") + (lane == nullptr ? "NULL" :  lane->getID()), lane, doAdd),
      64      9042550 :     myParent(parent),
      65      9042550 :     myLaneLength(length),
      66      9042550 :     sampleSeconds(0),
      67     27127650 :     travelledDistance(0) { }
      68              : 
      69              : 
      70      8971312 : MSMeanData::MeanDataValues::~MeanDataValues() {
      71      8971312 : }
      72              : 
      73              : 
      74              : bool
      75       690020 : 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       690020 :     return myParent == nullptr || myParent->vehicleApplies(veh);
      83              : }
      84              : 
      85              : 
      86              : bool
      87    458447370 : 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    458447370 :     const double oldSpeed = veh.getPreviousSpeed();
      91    458447370 :     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    458447370 :     double timeOnLane = TS;
      96    458447370 :     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    458447370 :     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    458447370 :     if (oldPos < 0 && newPos >= 0) {
     107              :         // Vehicle was not on this lane in the last time step
     108     10089454 :         timeBeforeEnter = MSCFModel::passingTime(oldPos, 0, newPos, oldSpeed, newSpeed);
     109     10089454 :         timeOnLane = TS - timeBeforeEnter;
     110              :         frontOnLane = timeOnLane;
     111     10089454 :         enterSpeed = MSCFModel::speedAfterTime(timeBeforeEnter, oldSpeed, newPos - oldPos);
     112              :     }
     113              : 
     114    458447370 :     const double oldBackPos = oldPos - veh.getVehicleType().getLength();
     115    458447370 :     const double newBackPos = newPos - veh.getVehicleType().getLength();
     116              : 
     117              :     // Determine the time before the vehicle back enters
     118    458447370 :     if (oldBackPos < 0. && newBackPos > 0.) {
     119     10340947 :         timeBeforeEnterBack = MSCFModel::passingTime(oldBackPos, 0., newBackPos, oldSpeed, newSpeed);
     120    448106423 :     } else if (newBackPos <= 0) {
     121      6742483 :         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    458447370 :     if (newBackPos > myLaneLength // vehicle's back has left the lane
     128     11555746 :             && 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     11555746 :         timeBeforeLeave = MSCFModel::passingTime(oldBackPos, myLaneLength, newBackPos, oldSpeed, newSpeed);
     132     11555746 :         const double timeAfterLeave = TS - timeBeforeLeave;
     133     11555746 :         timeOnLane -= timeAfterLeave;
     134     11555746 :         leaveSpeed = MSCFModel::speedAfterTime(timeBeforeLeave, oldSpeed, newPos - oldPos);
     135              :         // XXX: Do we really need this? Why would this "reduce rounding errors"? (Leo) Refs. #2579
     136     11555746 :         if (fabs(timeOnLane) < NUMERICAL_EPS) { // reduce rounding errors
     137              :             timeOnLane = 0.;
     138              :         }
     139     11555746 :         ret = veh.hasArrived();
     140              :     }
     141              : 
     142              :     // Treat the case that the vehicle's front left the lane in the last step
     143    458447370 :     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     12569847 :         timeBeforeLeaveFront = MSCFModel::passingTime(oldPos, myLaneLength, newPos, oldSpeed, newSpeed);
     147     12569847 :         const double timeAfterLeave = TS - timeBeforeLeaveFront;
     148     12569847 :         frontOnLane -= timeAfterLeave;
     149              :         // XXX: Do we really need this? Why would this "reduce rounding errors"? (Leo) Refs. #2579
     150     12569847 :         if (fabs(frontOnLane) < NUMERICAL_EPS) { // reduce rounding errors
     151              :             frontOnLane = 0.;
     152              :         }
     153     12569847 :         leaveSpeedFront = MSCFModel::speedAfterTime(timeBeforeLeaveFront, oldSpeed, newPos - oldPos);
     154              :     }
     155              : 
     156              :     assert(frontOnLane <= TS);
     157              :     assert(timeOnLane <= TS);
     158              : 
     159    458447370 :     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    458447370 :     if (timeOnLane == 0) {
     164        20745 :         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    458426625 :     double vehLength = veh.getVehicleType().getLength();
     192              :     // occupied lane length at timeBeforeEnter (resp. stepStart if already on lane)
     193    458426625 :     double lengthOnLaneAtStepStart = MAX2(0., MIN4(myLaneLength, vehLength, vehLength - (oldPos - myLaneLength), oldPos));
     194              :     // occupied lane length at timeBeforeLeave (resp. stepEnd if still on lane)
     195    458426625 :     double lengthOnLaneAtStepEnd = MAX2(0., MIN4(myLaneLength, vehLength, vehLength - (newPos - myLaneLength), newPos));
     196              :     double integratedLengthOnLane = 0.;
     197    458426625 :     if (timeBeforeEnterBack < timeBeforeLeaveFront) {
     198              :         // => timeBeforeLeaveFront>0, myLaneLength>vehLength
     199              :         // vehicle length on detector at timeBeforeEnterBack
     200    445455609 :         double lengthOnLaneAtBackEnter = MIN2(veh.getVehicleType().getLength(), newPos);
     201              :         // linear quadrature of occupancy between timeBeforeEnter and timeBeforeEnterBack
     202    445455609 :         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    445455609 :         integratedLengthOnLane += (timeBeforeLeaveFront - timeBeforeEnterBack) * vehLength;
     206              :         // and until vehicle leaves/stepEnd
     207    445455609 :         integratedLengthOnLane += (timeBeforeLeave - timeBeforeLeaveFront) * (vehLength + lengthOnLaneAtStepEnd) * 0.5;
     208     12971016 :     } else if (timeBeforeEnterBack >= timeBeforeLeaveFront) {
     209              :         // => myLaneLength <= vehLength or (timeBeforeLeaveFront == timeBeforeEnterBack == 0)
     210              :         // vehicle length on detector at timeBeforeLeaveFront
     211              :         double lengthOnLaneAtLeaveFront;
     212     12971016 :         if (timeBeforeLeaveFront == timeBeforeEnter) {
     213              :             // for the case that front already left
     214              :             lengthOnLaneAtLeaveFront = lengthOnLaneAtStepStart;
     215      6766480 :         } 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     12971016 :         integratedLengthOnLane += (timeBeforeLeaveFront - timeBeforeEnter) * (lengthOnLaneAtLeaveFront + lengthOnLaneAtStepStart) * 0.5;
     226              :         // linear quadrature of occupancy between timeBeforeLeaveFront and timeBeforeEnterBack
     227     12971016 :         integratedLengthOnLane += (timeBeforeEnterBack - timeBeforeLeaveFront) * lengthOnLaneAtLeaveFront;
     228              :         // and until vehicle leaves/stepEnd
     229     12971016 :         integratedLengthOnLane += (timeBeforeLeave - timeBeforeEnterBack) * (lengthOnLaneAtLeaveFront + lengthOnLaneAtStepEnd) * 0.5;
     230              :     }
     231              : 
     232    458426625 :     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    458426625 :     const double travelledDistanceFrontOnLane = MSGlobals::gSemiImplicitEulerUpdate ? frontOnLane * newSpeed
     244    162223496 :             : MAX2(0., MIN2(newPos, myLaneLength) - MAX2(oldPos, 0.));
     245    539538373 :     const double travelledDistanceVehicleOnLane = MSGlobals::gSemiImplicitEulerUpdate ? timeOnLane * newSpeed
     246    162223496 :             : 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    458426625 :     ScopedLocker<> lock(myNotificationMutex, MSGlobals::gNumSimThreads > 1);
     253              : #endif
     254    458426625 :     notifyMoveInternal(veh, frontOnLane, timeOnLane, (enterSpeed + leaveSpeedFront) / 2., (enterSpeed + leaveSpeed) / 2., travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
     255              :     return ret;
     256              : }
     257              : 
     258              : 
     259              : bool
     260      1179296 : MSMeanData::MeanDataValues::notifyLeave(SUMOTrafficObject& /*veh*/, double /*lastPos*/, MSMoveReminder::Notification reason, const MSLane* /* enteredLane */) {
     261      1179296 :     if (MSGlobals::gUseMesoSim) {
     262              :         return false; // reminder is re-added on every segment (@recheck for performance)
     263              :     }
     264      1024768 :     return reason == MSMoveReminder::NOTIFICATION_JUNCTION;
     265              : }
     266              : 
     267              : 
     268              : bool
     269      2951396 : MSMeanData::MeanDataValues::isEmpty() const {
     270      2951396 :     return sampleSeconds == 0;
     271              : }
     272              : 
     273              : 
     274              : void
     275            0 : MSMeanData::MeanDataValues::update() {
     276            0 : }
     277              : 
     278              : 
     279              : double
     280       967794 : MSMeanData::MeanDataValues::getSamples() const {
     281       967794 :     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(new TrackerEntry(parent->createValues(lane, length, false)));
     293         2744 : }
     294              : 
     295              : 
     296         5488 : MSMeanData::MeanDataValueTracker::~MeanDataValueTracker() {
     297              :     std::list<TrackerEntry*>::iterator i;
     298        29072 :     for (i = myCurrentData.begin(); i != myCurrentData.end(); i++) {
     299        26328 :         delete *i;
     300              :     }
     301              : 
     302              :     // FIXME: myTrackedData may still hold some undeleted TrackerEntries. When to delete those? (Leo), refers to #2251
     303              :     // code below fails
     304              : 
     305              : //      std::map<SUMOTrafficObject*, TrackerEntry*>::iterator j;
     306              : //      for(j=myTrackedData.begin(); j!=myTrackedData.end();j++){
     307              : //              delete j->second;
     308              : //      }
     309         5488 : }
     310              : 
     311              : 
     312              : void
     313       164304 : MSMeanData::MeanDataValueTracker::reset(bool afterWrite) {
     314       164304 :     if (afterWrite) {
     315        70360 :         if (myCurrentData.begin() != myCurrentData.end()) {
     316              :             // delete myCurrentData.front();
     317        70360 :             myCurrentData.pop_front();
     318              :         }
     319              :     } else {
     320        93944 :         myCurrentData.push_back(new TrackerEntry(myParent->createValues(myLane, myLaneLength, false)));
     321              :     }
     322       164304 : }
     323              : 
     324              : 
     325              : void
     326            0 : MSMeanData::MeanDataValueTracker::addTo(MSMeanData::MeanDataValues& val) const {
     327            0 :     myCurrentData.front()->myValues->addTo(val);
     328            0 : }
     329              : 
     330              : 
     331              : void
     332        15042 : 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) {
     333        15042 :     myTrackedData[&veh]->myValues->notifyMoveInternal(veh, frontOnLane, timeOnLane, meanSpeedFrontOnLane, meanSpeedVehicleOnLane, travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
     334        15042 : }
     335              : 
     336              : 
     337              : bool
     338          787 : MSMeanData::MeanDataValueTracker::notifyLeave(SUMOTrafficObject& veh, double lastPos, MSMoveReminder::Notification reason, const MSLane* /* enteredLane */) {
     339          787 :     if (myParent == nullptr || reason != MSMoveReminder::NOTIFICATION_SEGMENT) {
     340          387 :         myTrackedData[&veh]->myNumVehicleLeft++;
     341              :     }
     342          787 :     return myTrackedData[&veh]->myValues->notifyLeave(veh, lastPos, reason);
     343              : }
     344              : 
     345              : 
     346              : bool
     347         3400 : MSMeanData::MeanDataValueTracker::notifyEnter(SUMOTrafficObject& veh, MSMoveReminder::Notification reason, const MSLane* enteredLane) {
     348              : #ifdef DEBUG_NOTIFY_ENTER
     349              :     std::cout << "\n" << SIMTIME << " MSMeanData::MeanDataValueTracker: veh '" << veh.getID() << "' enters lane '" << enteredLane->getID() << "'" << std::endl;
     350              : #else
     351              :     UNUSED_PARAMETER(enteredLane);
     352              : #endif
     353         3400 :     if (reason == MSMoveReminder::NOTIFICATION_SEGMENT) {
     354              :         return true;
     355              :     }
     356         5768 :     if (myParent->vehicleApplies(veh) && myTrackedData.find(&veh) == myTrackedData.end()) {
     357          507 :         myTrackedData[&veh] = myCurrentData.back();
     358          507 :         myTrackedData[&veh]->myNumVehicleEntered++;
     359          507 :         if (!myTrackedData[&veh]->myValues->notifyEnter(veh, reason)) {
     360            0 :             myTrackedData[&veh]->myNumVehicleLeft++;
     361            0 :             myTrackedData.erase(&veh);
     362            0 :             return false;
     363              :         }
     364              :         return true;
     365              :     }
     366              :     return false;
     367              : }
     368              : 
     369              : 
     370              : bool
     371        91696 : MSMeanData::MeanDataValueTracker::isEmpty() const {
     372        91696 :     return myCurrentData.front()->myValues->isEmpty();
     373              : }
     374              : 
     375              : 
     376              : void
     377          368 : MSMeanData::MeanDataValueTracker::write(OutputDevice& dev,
     378              :                                         const SumoXMLAttrMask& attributeMask,
     379              :                                         const SUMOTime period,
     380              :                                         const int numLanes,
     381              :                                         const double speedLimit,
     382              :                                         const double defaultTravelTime,
     383              :                                         const int /*numVehicles*/) const {
     384          368 :     myCurrentData.front()->myValues->write(dev, attributeMask, period, numLanes, speedLimit,
     385              :                                            defaultTravelTime,
     386              :                                            myCurrentData.front()->myNumVehicleEntered);
     387          368 : }
     388              : 
     389              : 
     390              : int
     391        63116 : MSMeanData::MeanDataValueTracker::getNumReady() const {
     392              :     int result = 0;
     393      1445056 :     for (std::list<TrackerEntry*>::const_iterator it = myCurrentData.begin(); it != myCurrentData.end(); ++it) {
     394      1384592 :         if ((*it)->myNumVehicleEntered == (*it)->myNumVehicleLeft) {
     395      1381940 :             result++;
     396              :         } else {
     397              :             break;
     398              :         }
     399              :     }
     400        63116 :     return result;
     401              : }
     402              : 
     403              : 
     404              : double
     405          368 : MSMeanData::MeanDataValueTracker::getSamples() const {
     406          368 :     return myCurrentData.front()->myValues->getSamples();
     407              : }
     408              : 
     409              : 
     410              : // ---------------------------------------------------------------------------
     411              : // MSMeanData - methods
     412              : // ---------------------------------------------------------------------------
     413        21363 : MSMeanData::MSMeanData(const std::string& id,
     414              :                        const SUMOTime dumpBegin, const SUMOTime dumpEnd,
     415              :                        const bool useLanes, const bool withEmpty,
     416              :                        const bool printDefaults, const bool withInternal,
     417              :                        const bool trackVehicles,
     418              :                        const int detectPersons,
     419              :                        const double maxTravelTime,
     420              :                        const double minSamples,
     421              :                        const std::string& vTypes,
     422              :                        const std::string& writeAttributes,
     423              :                        const std::vector<MSEdge*>& edges,
     424        21363 :                        bool aggregate) :
     425              :     MSDetectorFileOutput(id, vTypes, "", detectPersons),
     426        21363 :     myMinSamples(minSamples),
     427        21363 :     myMaxTravelTime(maxTravelTime),
     428        21363 :     myDumpEmpty(withEmpty),
     429        21363 :     myAmEdgeBased(!useLanes),
     430        21363 :     myDumpBegin(dumpBegin),
     431        21363 :     myDumpEnd(dumpEnd),
     432        21363 :     myInitTime(SUMOTime_MAX),
     433        21363 :     myEdges(edges),
     434        21363 :     myPrintDefaults(printDefaults),
     435        21363 :     myDumpInternal(withInternal && MSGlobals::gUsingInternalLanes),
     436        21363 :     myTrackVehicles(trackVehicles),
     437        85452 :     myWrittenAttributes(OutputDevice::parseWrittenAttributes(StringTokenizer(writeAttributes).getVector(), "meandata '" + id + "'")),
     438        64089 :     myAggregate(aggregate)
     439        21363 : { }
     440              : 
     441              : 
     442              : void
     443        20229 : MSMeanData::init() {
     444        20229 :     myInitTime = MSNet::getInstance()->getCurrentTimeStep();
     445        20229 :     if (myEdges.empty()) {
     446              :         // use all edges by default
     447       706441 :         for (MSEdge* const edge : MSNet::getInstance()->getEdgeControl().getEdges()) {
     448       686277 :             if ((myDumpInternal || !edge->isInternal()) &&
     449       457692 :                     ((detectsPersons() && myDumpInternal) || (!edge->isCrossing() && !edge->isWalkingArea()))) {
     450       314778 :                 myEdges.push_back(edge);
     451              :             }
     452              :         }
     453              :     }
     454              :     int index = 0;
     455       335116 :     for (MSEdge* edge : myEdges) {
     456       314887 :         myMeasures.push_back(std::vector<MeanDataValues*>());
     457       314887 :         myEdgeIndex[edge] = index++;
     458       314887 :         const std::vector<MSLane*>& lanes = edge->getLanes();
     459       314887 :         if (MSGlobals::gUseMesoSim) {
     460              :             MeanDataValues* data;
     461        29093 :             if (!myAmEdgeBased) {
     462           85 :                 for (MSLane* const lane : lanes) {
     463           52 :                     data = createValues(lane, lanes[0]->getLength(), false);
     464           52 :                     myMeasures.back().push_back(data);
     465           52 :                     MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     466          310 :                     while (s != nullptr) {
     467          258 :                         s->addDetector(data, lane->getIndex());
     468          258 :                         s->prepareDetectorForWriting(*data, lane->getIndex());
     469              :                         s = s->getNextSegment();
     470              :                     }
     471           52 :                     data->reset();
     472           52 :                     data->reset(true);
     473              :                 }
     474              :             } else {
     475        29060 :                 if (myTrackVehicles) {
     476          880 :                     data = new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this);
     477              :                 } else {
     478        28180 :                     data = createValues(nullptr, lanes[0]->getLength(), false);
     479              :                 }
     480        58120 :                 data->setDescription("meandata_" + getID() + "|" + edge->getID());
     481        29060 :                 myMeasures.back().push_back(data);
     482        29060 :                 MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     483       148016 :                 while (s != nullptr) {
     484       118956 :                     s->addDetector(data);
     485       118956 :                     s->prepareDetectorForWriting(*data);
     486              :                     s = s->getNextSegment();
     487              :                 }
     488        29060 :                 data->reset();
     489        29060 :                 data->reset(true);
     490              :             }
     491              :             continue;
     492        29093 :         }
     493       285794 :         if (myAmEdgeBased && myTrackVehicles) {
     494          936 :             myMeasures.back().push_back(new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this));
     495              :         }
     496       740908 :         for (MSLane* const lane : lanes) {
     497       455114 :             if (myTrackVehicles) {
     498         2016 :                 if (myAmEdgeBased) {
     499         1088 :                     lane->addMoveReminder(myMeasures.back().back());
     500              :                 } else {
     501          928 :                     myMeasures.back().push_back(new MeanDataValueTracker(lane, lane->getLength(), this));
     502              :                 }
     503              :             } else {
     504       453098 :                 myMeasures.back().push_back(createValues(lane, lane->getLength(), true));
     505              :             }
     506              :         }
     507              :     }
     508        20229 : }
     509              : 
     510              : 
     511        21314 : MSMeanData::~MSMeanData() {
     512       335724 :     for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
     513       797607 :         for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
     514       483197 :             delete *j;
     515              :         }
     516              :     }
     517        21314 : }
     518              : 
     519              : 
     520              : void
     521        12645 : MSMeanData::resetOnly(SUMOTime stopTime) {
     522              :     UNUSED_PARAMETER(stopTime);
     523        12645 :     if (MSGlobals::gUseMesoSim) {
     524              :         MSEdgeVector::iterator edge = myEdges.begin();
     525       117300 :         for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i, ++edge) {
     526       113716 :             MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(**edge);
     527       227432 :             for (MeanDataValues* data : *i) {
     528       602292 :                 while (s != nullptr) {
     529       488576 :                     s->prepareDetectorForWriting(*data);
     530              :                     s = s->getNextSegment();
     531              :                 }
     532       113716 :                 data->reset();
     533              :             }
     534              :         }
     535              :         return;
     536              :     }
     537       257304 :     for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
     538       529990 :         for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
     539       281747 :             (*j)->reset();
     540              :         }
     541              :     }
     542              : }
     543              : 
     544              : 
     545              : std::string
     546      9814741 : MSMeanData::getEdgeID(const MSEdge* const edge) {
     547      9814741 :     return edge->getID();
     548              : }
     549              : 
     550              : 
     551              : void
     552          940 : MSMeanData::writeAggregated(OutputDevice& dev, SUMOTime startTime, SUMOTime stopTime) {
     553          940 :     if (myTrackVehicles) {
     554            0 :         throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
     555              :     }
     556              : 
     557              :     double edgeLengthSum = 0;
     558              :     int laneNumber = 0;
     559              :     double speedSum = 0;
     560              :     double totalTT = 0;
     561        42172 :     for (MSEdge* edge : myEdges) {
     562        41232 :         edgeLengthSum += edge->getLength();
     563        41232 :         laneNumber += edge->getNumDrivingLanes();
     564        41232 :         speedSum += edge->getSpeedLimit();
     565        41232 :         totalTT += edge->getLength() / edge->getSpeedLimit();
     566              :     }
     567          940 :     MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
     568        42172 :     for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
     569        82512 :         for (MeanDataValues* meanData : edgeValues) {
     570        41280 :             meanData->addTo(*sumData);
     571        41280 :             if (!MSNet::getInstance()->skipFinalReset()) {
     572        41280 :                 meanData->reset();
     573              :             }
     574              :         }
     575              :     }
     576          940 :     if (MSGlobals::gUseMesoSim) {
     577        14040 :         for (int i = 0; i < (int)myEdges.size(); i++) {
     578        13728 :             MSEdge* edge = myEdges[i];
     579              :             std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
     580        13728 :             MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     581        68640 :             while (s != nullptr) {
     582       109824 :                 for (MeanDataValues* meanData : edgeValues) {
     583        54912 :                     s->prepareDetectorForWriting(*meanData);
     584        54912 :                     meanData->addTo(*sumData);
     585        54912 :                     if (!MSNet::getInstance()->skipFinalReset()) {
     586        54912 :                         meanData->reset();
     587              :                     }
     588              :                 }
     589              :                 s = s->getNextSegment();
     590              :             }
     591              :         }
     592              :     }
     593              : 
     594         1880 :     if (writePrefix(dev, *sumData, SUMO_TAG_EDGE, "AGGREGATED")) {
     595          934 :         dev.writeAttr(SUMO_ATTR_NUMEDGES, myEdges.size());
     596          934 :         sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)myEdges.size(),
     597          934 :                        myPrintDefaults ? totalTT : -1.);
     598              :     }
     599          940 :     delete sumData;
     600          940 : }
     601              : 
     602              : 
     603              : void
     604     17848068 : MSMeanData::writeEdge(OutputDevice& dev,
     605              :                       const std::vector<MeanDataValues*>& edgeValues,
     606              :                       const MSEdge* const edge, SUMOTime startTime, SUMOTime stopTime) {
     607     17848068 :     if (MSGlobals::gUseMesoSim) {
     608              :         int idx = 0;
     609      2719305 :         for (MeanDataValues* const data : edgeValues) {
     610      1359676 :             MESegment* s = MSGlobals::gMesoNet->getSegmentForEdge(*edge);
     611      6992732 :             while (s != nullptr) {
     612      5633398 :                 s->prepareDetectorForWriting(*data, myAmEdgeBased ? -1 : idx);
     613              :                 s = s->getNextSegment();
     614              :             }
     615      1359676 :             idx++;
     616              :         }
     617      1359629 :         if (myAmEdgeBased) {
     618      1359582 :             MeanDataValues* data = edgeValues.front();
     619      2719164 :             if (writePrefix(dev, *data, SUMO_TAG_EDGE, getEdgeID(edge))) {
     620        59584 :                 data->write(dev, myWrittenAttributes, stopTime - startTime,
     621              :                             edge->getNumDrivingLanes(),
     622              :                             edge->getSpeedLimit(),
     623        59584 :                             myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
     624              :             }
     625      1359582 :             if (!MSNet::getInstance()->skipFinalReset()) {
     626      1359582 :                 data->reset(true);
     627              :             }
     628              :             return;
     629              :         }
     630              :     }
     631              :     std::vector<MeanDataValues*>::const_iterator lane;
     632     16488486 :     if (!myAmEdgeBased) {
     633      8007863 :         bool writeCheck = myDumpEmpty;
     634      8007863 :         if (!writeCheck) {
     635     15611257 :             for (lane = edgeValues.begin(); lane != edgeValues.end(); ++lane) {
     636      7912844 :                 if (!(*lane)->isEmpty()) {
     637              :                     writeCheck = true;
     638              :                     break;
     639              :                 }
     640              :             }
     641              :         }
     642      7901496 :         if (writeCheck) {
     643       309450 :             dev.openTag(SUMO_TAG_EDGE).writeAttr(SUMO_ATTR_ID, edge->getID());
     644              :         }
     645     16117028 :         for (lane = edgeValues.begin(); lane != edgeValues.end(); ++lane) {
     646      8109165 :             MeanDataValues& meanData = **lane;
     647     24327495 :             if (writePrefix(dev, meanData, SUMO_TAG_LANE, meanData.getLane()->getID())) {
     648       399338 :                 meanData.write(dev, myWrittenAttributes, stopTime - startTime, 1, meanData.getLane()->getSpeedLimit(),
     649       399338 :                                myPrintDefaults ? meanData.getLane()->getLength() / meanData.getLane()->getSpeedLimit() : -1.);
     650              :             }
     651      8109165 :             if (!MSNet::getInstance()->skipFinalReset()) {
     652      8109165 :                 meanData.reset(true);
     653              :             }
     654              :         }
     655      8007863 :         if (writeCheck) {
     656       618900 :             dev.closeTag();
     657              :         }
     658              :     } else {
     659      8480623 :         if (myTrackVehicles) {
     660        22440 :             MeanDataValues& meanData = **edgeValues.begin();
     661        67320 :             if (writePrefix(dev, meanData, SUMO_TAG_EDGE, edge->getID())) {
     662          160 :                 meanData.write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
     663          160 :                                myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
     664              :             }
     665        22440 :             if (!MSNet::getInstance()->skipFinalReset()) {
     666        22440 :                 meanData.reset(true);
     667              :             }
     668              :         } else {
     669      8458183 :             MeanDataValues* sumData = createValues(nullptr, edge->getLength(), false);
     670     17065320 :             for (lane = edgeValues.begin(); lane != edgeValues.end(); ++lane) {
     671      8607137 :                 MeanDataValues& meanData = **lane;
     672      8607137 :                 meanData.addTo(*sumData);
     673      8607137 :                 if (!MSNet::getInstance()->skipFinalReset()) {
     674      8607137 :                     meanData.reset();
     675              :                 }
     676              :             }
     677     16916366 :             if (writePrefix(dev, *sumData, SUMO_TAG_EDGE, getEdgeID(edge))) {
     678       499392 :                 sumData->write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
     679       499392 :                                myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
     680              :             }
     681      8458182 :             delete sumData;
     682              :         }
     683              :     }
     684              : }
     685              : 
     686              : 
     687              : void
     688       434210 : MSMeanData::openInterval(OutputDevice& dev, const SUMOTime startTime, const SUMOTime stopTime) {
     689       868420 :     dev.openTag(SUMO_TAG_INTERVAL).writeAttr(SUMO_ATTR_BEGIN, time2string(startTime)).writeAttr(SUMO_ATTR_END, time2string(stopTime));
     690       434210 :     dev.writeAttr(SUMO_ATTR_ID, myID);
     691       434210 : }
     692              : 
     693              : 
     694              : bool
     695     17947286 : MSMeanData::writePrefix(OutputDevice& dev, const MeanDataValues& values, const SumoXMLTag tag, const std::string id) const {
     696     17947286 :     if (myDumpEmpty || !values.isEmpty()) {
     697       956688 :         dev.openTag(tag);
     698       956688 :         dev.writeAttr(SUMO_ATTR_ID, id);
     699       956688 :         dev.writeOptionalAttr(SUMO_ATTR_SAMPLEDSECONDS, values.getSamples(), myWrittenAttributes);
     700       956688 :         return true;
     701              :     }
     702              :     return false;
     703              : }
     704              : 
     705              : 
     706              : void
     707       445519 : MSMeanData::writeXMLOutput(OutputDevice& dev,
     708              :                            SUMOTime startTime, SUMOTime stopTime) {
     709              :     // check whether this dump shall be written for the current time
     710       445519 :     int numReady = myDumpBegin < stopTime && myDumpEnd - DELTA_T >= startTime ? 1 : 0;
     711       445519 :     if (myTrackVehicles && myDumpBegin < stopTime) {
     712         2124 :         myPendingIntervals.push_back(std::make_pair(startTime, stopTime));
     713         2124 :         numReady = (int)myPendingIntervals.size();
     714        63312 :         for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
     715       124304 :             for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
     716        63116 :                 numReady = MIN2(numReady, ((MeanDataValueTracker*)*j)->getNumReady());
     717        63116 :                 if (numReady == 0) {
     718              :                     break;
     719              :                 }
     720              :             }
     721        63092 :             if (numReady == 0) {
     722              :                 break;
     723              :             }
     724              :         }
     725              :     }
     726       445519 :     const bool partialInterval = startTime < myInitTime;
     727       445519 :     if (numReady == 0 || myTrackVehicles || partialInterval) {
     728        12645 :         resetOnly(stopTime);
     729              :     }
     730       445519 :     if (partialInterval) {
     731              :         return;
     732              :     }
     733       879970 :     while (numReady-- > 0) {
     734       434462 :         if (!myPendingIntervals.empty()) {
     735         1588 :             startTime = myPendingIntervals.front().first;
     736         1588 :             stopTime = myPendingIntervals.front().second;
     737         1588 :             myPendingIntervals.pop_front();
     738              :         }
     739       434462 :         openInterval(dev, startTime, stopTime);
     740       434462 :         if (myAggregate) {
     741          940 :             writeAggregated(dev, startTime, stopTime);
     742              :         } else {
     743              :             MSEdgeVector::const_iterator edge = myEdges.begin();
     744     18281589 :             for (const std::vector<MeanDataValues*>& measures : myMeasures) {
     745     17848068 :                 writeEdge(dev, measures, *edge, startTime, stopTime);
     746              :                 ++edge;
     747              :             }
     748              :         }
     749       868922 :         dev.closeTag();
     750              :     }
     751              :     dev.flush();
     752              : }
     753              : 
     754              : 
     755              : void
     756        20209 : MSMeanData::writeXMLDetectorProlog(OutputDevice& dev) const {
     757        40418 :     dev.writeXMLHeader("meandata", "meandata_file.xsd");
     758        20209 : }
     759              : 
     760              : 
     761              : void
     762     63140866 : MSMeanData::detectorUpdate(const SUMOTime step) {
     763     63140866 :     if (step + DELTA_T == myDumpBegin) {
     764          639 :         init();
     765              :     }
     766     63140866 : }
     767              : 
     768              : 
     769              : const std::vector<MSMeanData::MeanDataValues*>*
     770            0 : MSMeanData::getEdgeValues(const MSEdge* edge) const {
     771              :     auto it = myEdgeIndex.find(edge);
     772            0 :     if (it != myEdgeIndex.end()) {
     773            0 :         return &myMeasures[it->second];
     774              :     } else {
     775              :         return nullptr;
     776              :     }
     777              : }
     778              : 
     779              : 
     780              : const std::vector<MSMoveReminder*>
     781            1 : MSMeanData::getReminders() const {
     782              :     std::vector<MSMoveReminder*> result;
     783           45 :     for (auto vec : myMeasures) {
     784           44 :         result.insert(result.end(), vec.begin(), vec.end());
     785           44 :     }
     786            1 :     return result;
     787            0 : }
     788              : 
     789              : 
     790              : /****************************************************************************/
        

Generated by: LCOV version 2.0-1