Eclipse SUMO - Simulation of Urban MObility
Loading...
Searching...
No Matches
MSMeanData.cpp
Go to the documentation of this file.
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/****************************************************************************/
22// Data collector for edges/lanes
23/****************************************************************************/
24#include <config.h>
25
26#include <limits>
27#ifdef HAVE_FOX
29#endif
35#include <microsim/MSEdge.h>
36#include <microsim/MSLane.h>
37#include <microsim/MSVehicle.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// ---------------------------------------------------------------------------
61 MSLane* const lane, const double length, const bool doAdd,
62 const MSMeanData* const parent) :
63 MSMoveReminder("meandata_" + (parent == nullptr ? "" : parent->getID() + "|") + (lane == nullptr ? "NULL" : lane->getID()), lane, doAdd),
64 myParent(parent),
65 myLaneLength(length),
66 sampleSeconds(0),
67 travelledDistance(0) { }
68
69
72
73
74bool
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 return myParent == nullptr || myParent->vehicleApplies(veh);
83}
84
85
86bool
87MSMeanData::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 const double oldSpeed = veh.getPreviousSpeed();
91 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 double timeOnLane = TS;
96 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 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 if (oldPos < 0 && newPos >= 0) {
107 // Vehicle was not on this lane in the last time step
108 timeBeforeEnter = MSCFModel::passingTime(oldPos, 0, newPos, oldSpeed, newSpeed);
109 timeOnLane = TS - timeBeforeEnter;
110 frontOnLane = timeOnLane;
111 enterSpeed = MSCFModel::speedAfterTime(timeBeforeEnter, oldSpeed, newPos - oldPos);
112 }
113
114 const double oldBackPos = oldPos - veh.getVehicleType().getLength();
115 const double newBackPos = newPos - veh.getVehicleType().getLength();
116
117 // Determine the time before the vehicle back enters
118 if (oldBackPos < 0. && newBackPos > 0.) {
119 timeBeforeEnterBack = MSCFModel::passingTime(oldBackPos, 0., newBackPos, oldSpeed, newSpeed);
120 } else if (newBackPos <= 0) {
121 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 if (newBackPos > myLaneLength // vehicle's back has left the lane
128 && 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 timeBeforeLeave = MSCFModel::passingTime(oldBackPos, myLaneLength, newBackPos, oldSpeed, newSpeed);
132 const double timeAfterLeave = TS - timeBeforeLeave;
133 timeOnLane -= timeAfterLeave;
134 leaveSpeed = MSCFModel::speedAfterTime(timeBeforeLeave, oldSpeed, newPos - oldPos);
135 // XXX: Do we really need this? Why would this "reduce rounding errors"? (Leo) Refs. #2579
136 if (fabs(timeOnLane) < NUMERICAL_EPS) { // reduce rounding errors
137 timeOnLane = 0.;
138 }
139 ret = veh.hasArrived();
140 }
141
142 // Treat the case that the vehicle's front left the lane in the last step
143 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 timeBeforeLeaveFront = MSCFModel::passingTime(oldPos, myLaneLength, newPos, oldSpeed, newSpeed);
147 const double timeAfterLeave = TS - timeBeforeLeaveFront;
148 frontOnLane -= timeAfterLeave;
149 // XXX: Do we really need this? Why would this "reduce rounding errors"? (Leo) Refs. #2579
150 if (fabs(frontOnLane) < NUMERICAL_EPS) { // reduce rounding errors
151 frontOnLane = 0.;
152 }
153 leaveSpeedFront = MSCFModel::speedAfterTime(timeBeforeLeaveFront, oldSpeed, newPos - oldPos);
154 }
155
156 assert(frontOnLane <= TS);
157 assert(timeOnLane <= TS);
158
159 if (timeOnLane < 0) {
160 WRITE_ERRORF(TL("Negative vehicle step fraction for '%' on lane '%'."), veh.getID(), getLane()->getID());
161 return veh.hasArrived();
162 }
163 if (timeOnLane == 0) {
164 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 double vehLength = veh.getVehicleType().getLength();
192 // occupied lane length at timeBeforeEnter (resp. stepStart if already on lane)
193 double lengthOnLaneAtStepStart = MAX2(0., MIN4(myLaneLength, vehLength, vehLength - (oldPos - myLaneLength), oldPos));
194 // occupied lane length at timeBeforeLeave (resp. stepEnd if still on lane)
195 double lengthOnLaneAtStepEnd = MAX2(0., MIN4(myLaneLength, vehLength, vehLength - (newPos - myLaneLength), newPos));
196 double integratedLengthOnLane = 0.;
197 if (timeBeforeEnterBack < timeBeforeLeaveFront) {
198 // => timeBeforeLeaveFront>0, myLaneLength>vehLength
199 // vehicle length on detector at timeBeforeEnterBack
200 double lengthOnLaneAtBackEnter = MIN2(veh.getVehicleType().getLength(), newPos);
201 // linear quadrature of occupancy between timeBeforeEnter and timeBeforeEnterBack
202 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 integratedLengthOnLane += (timeBeforeLeaveFront - timeBeforeEnterBack) * vehLength;
206 // and until vehicle leaves/stepEnd
207 integratedLengthOnLane += (timeBeforeLeave - timeBeforeLeaveFront) * (vehLength + lengthOnLaneAtStepEnd) * 0.5;
208 } else if (timeBeforeEnterBack >= timeBeforeLeaveFront) {
209 // => myLaneLength <= vehLength or (timeBeforeLeaveFront == timeBeforeEnterBack == 0)
210 // vehicle length on detector at timeBeforeLeaveFront
211 double lengthOnLaneAtLeaveFront;
212 if (timeBeforeLeaveFront == timeBeforeEnter) {
213 // for the case that front already left
214 lengthOnLaneAtLeaveFront = lengthOnLaneAtStepStart;
215 } 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 integratedLengthOnLane += (timeBeforeLeaveFront - timeBeforeEnter) * (lengthOnLaneAtLeaveFront + lengthOnLaneAtStepStart) * 0.5;
226 // linear quadrature of occupancy between timeBeforeLeaveFront and timeBeforeEnterBack
227 integratedLengthOnLane += (timeBeforeEnterBack - timeBeforeLeaveFront) * lengthOnLaneAtLeaveFront;
228 // and until vehicle leaves/stepEnd
229 integratedLengthOnLane += (timeBeforeLeave - timeBeforeEnterBack) * (lengthOnLaneAtLeaveFront + lengthOnLaneAtStepEnd) * 0.5;
230 }
231
232 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 const double travelledDistanceFrontOnLane = MSGlobals::gSemiImplicitEulerUpdate ? frontOnLane * newSpeed
244 : MAX2(0., MIN2(newPos, myLaneLength) - MAX2(oldPos, 0.));
245 const double travelledDistanceVehicleOnLane = MSGlobals::gSemiImplicitEulerUpdate ? timeOnLane * newSpeed
246 : 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 ScopedLocker<> lock(myNotificationMutex, MSGlobals::gNumSimThreads > 1);
253#endif
254 notifyMoveInternal(veh, frontOnLane, timeOnLane, (enterSpeed + leaveSpeedFront) / 2., (enterSpeed + leaveSpeed) / 2., travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
255 return ret;
256}
257
258
259bool
260MSMeanData::MeanDataValues::notifyLeave(SUMOTrafficObject& /*veh*/, double /*lastPos*/, MSMoveReminder::Notification reason, const MSLane* /* enteredLane */) {
262 return false; // reminder is re-added on every segment (@recheck for performance)
263 }
265}
266
267
268bool
270 return sampleSeconds == 0;
271}
272
273
274void
277
278
279double
281 return sampleSeconds;
282}
283
284
285// ---------------------------------------------------------------------------
286// MSMeanData::MeanDataValueTracker - methods
287// ---------------------------------------------------------------------------
289 const double length,
290 const MSMeanData* const parent)
291 : MSMeanData::MeanDataValues(lane, length, true, parent) {
292 myCurrentData.push_back(new TrackerEntry(parent->createValues(lane, length, false)));
293}
294
295
297 std::list<TrackerEntry*>::iterator i;
298 for (i = myCurrentData.begin(); i != myCurrentData.end(); i++) {
299 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}
310
311
312void
314 if (afterWrite) {
315 if (myCurrentData.begin() != myCurrentData.end()) {
316 // delete myCurrentData.front();
317 myCurrentData.pop_front();
318 }
319 } else {
320 myCurrentData.push_back(new TrackerEntry(myParent->createValues(myLane, myLaneLength, false)));
321 }
322}
323
324
325void
327 myCurrentData.front()->myValues->addTo(val);
328}
329
330
331void
332MSMeanData::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 myTrackedData[&veh]->myValues->notifyMoveInternal(veh, frontOnLane, timeOnLane, meanSpeedFrontOnLane, meanSpeedVehicleOnLane, travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
334}
335
336
337bool
339 if (myParent == nullptr || reason != MSMoveReminder::NOTIFICATION_SEGMENT) {
340 myTrackedData[&veh]->myNumVehicleLeft++;
341 }
342 return myTrackedData[&veh]->myValues->notifyLeave(veh, lastPos, reason);
343}
344
345
346bool
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
354 return true;
355 }
356 if (myParent->vehicleApplies(veh) && myTrackedData.find(&veh) == myTrackedData.end()) {
357 myTrackedData[&veh] = myCurrentData.back();
358 myTrackedData[&veh]->myNumVehicleEntered++;
359 if (!myTrackedData[&veh]->myValues->notifyEnter(veh, reason)) {
360 myTrackedData[&veh]->myNumVehicleLeft++;
361 myTrackedData.erase(&veh);
362 return false;
363 }
364 return true;
365 }
366 return false;
367}
368
369
370bool
372 return myCurrentData.front()->myValues->isEmpty();
373}
374
375
376void
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 myCurrentData.front()->myValues->write(dev, attributeMask, period, numLanes, speedLimit,
385 defaultTravelTime,
386 myCurrentData.front()->myNumVehicleEntered);
387}
388
389
390int
392 int result = 0;
393 for (std::list<TrackerEntry*>::const_iterator it = myCurrentData.begin(); it != myCurrentData.end(); ++it) {
394 if ((*it)->myNumVehicleEntered == (*it)->myNumVehicleLeft) {
395 result++;
396 } else {
397 break;
398 }
399 }
400 return result;
401}
402
403
404double
406 return myCurrentData.front()->myValues->getSamples();
407}
408
409
410// ---------------------------------------------------------------------------
411// MSMeanData - methods
412// ---------------------------------------------------------------------------
413MSMeanData::MSMeanData(const std::string& id,
414 const SUMOTime dumpBegin, const SUMOTime dumpEnd,
415 const bool useLanes, const std::string& excludeEmpty, const bool withInternal,
416 const bool trackVehicles,
417 const int detectPersons,
418 const double maxTravelTime,
419 const double minSamples,
420 const std::string& vTypes,
421 const std::string& writeAttributes,
422 const std::vector<MSEdge*>& edges,
423 AggregateType aggregate) :
424 MSDetectorFileOutput(id, vTypes, "", detectPersons),
425 myMinSamples(minSamples),
426 myMaxTravelTime(maxTravelTime),
427 myAmEdgeBased(!useLanes),
428 myDumpBegin(dumpBegin),
429 myDumpEnd(dumpEnd),
431 myEdges(edges),
432 myDumpInternal(withInternal && MSGlobals::gUsingInternalLanes),
433 myTrackVehicles(trackVehicles),
434 myWrittenAttributes(OutputDevice::parseWrittenAttributes(StringTokenizer(writeAttributes).getVector(), "meandata '" + id + "'")),
435 myAggregate(aggregate) {
436 try {
437 myDumpEmpty = !StringUtils::toBool(excludeEmpty);
438 } catch (const BoolFormatException&) {
439 if (excludeEmpty == "default" || excludeEmpty == "defaults") {
440 myPrintDefaults = true;
441 } else if (excludeEmpty == "modified") {
442 myDumpEmpty = false;
443 myPrintDefaults = true;
444 myPrintModified = true;
445 } else {
446 throw;
447 }
448 }
449}
450
451
452void
455 if (myEdges.empty()) {
456 // use all edges by default
457 for (MSEdge* const edge : MSNet::getInstance()->getEdgeControl().getEdges()) {
458 if ((myDumpInternal || !edge->isInternal()) &&
459 ((detectsPersons() && myDumpInternal) || (!edge->isCrossing() && !edge->isWalkingArea()))) {
460 myEdges.push_back(edge);
461 }
462 }
463 }
464 int index = 0;
465 for (MSEdge* edge : myEdges) {
466 myMeasures.push_back(std::vector<MeanDataValues*>());
467 myEdgeIndex[edge] = index++;
468 const std::vector<MSLane*>& lanes = edge->getLanes();
470 MeanDataValues* data;
471 if (!myAmEdgeBased) {
472 for (MSLane* const lane : lanes) {
473 data = createValues(lane, lanes[0]->getLength(), false);
474 myMeasures.back().push_back(data);
476 while (s != nullptr) {
477 s->addDetector(data, lane->getIndex());
478 s->prepareDetectorForWriting(*data, lane->getIndex());
479 s = s->getNextSegment();
480 }
481 data->reset();
482 data->reset(true);
483 }
484 } else {
485 if (myTrackVehicles) {
486 data = new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this);
487 } else {
488 data = createValues(nullptr, lanes[0]->getLength(), false);
489 }
490 data->setDescription("meandata_" + getID() + "|" + edge->getID());
491 myMeasures.back().push_back(data);
493 while (s != nullptr) {
494 s->addDetector(data);
496 s = s->getNextSegment();
497 }
498 data->reset();
499 data->reset(true);
500 }
501 continue;
502 }
504 myMeasures.back().push_back(new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this));
505 }
506 for (MSLane* const lane : lanes) {
507 if (myTrackVehicles) {
508 if (myAmEdgeBased) {
509 lane->addMoveReminder(myMeasures.back().back());
510 } else {
511 myMeasures.back().push_back(new MeanDataValueTracker(lane, lane->getLength(), this));
512 }
513 } else {
514 myMeasures.back().push_back(createValues(lane, lane->getLength(), true));
515 }
516 }
517 }
519 for (const MSEdge* e : MSEdge::getAllEdges()) {
520 if (e->isTazConnector()) {
521 myTAZ.push_back(e);
522 }
523 }
524 }
525}
526
527
529 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
530 for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
531 delete *j;
532 }
533 }
534}
535
536
537void
539 UNUSED_PARAMETER(stopTime);
541 MSEdgeVector::iterator edge = myEdges.begin();
542 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i, ++edge) {
544 for (MeanDataValues* data : *i) {
545 while (s != nullptr) {
547 s = s->getNextSegment();
548 }
549 data->reset();
550 }
551 }
552 return;
553 }
554 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
555 for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
556 (*j)->reset();
557 }
558 }
559}
560
561
562std::string
563MSMeanData::getEdgeID(const MSEdge* const edge) {
564 return edge->getID();
565}
566
567
568void
570 if (myTrackVehicles) {
571 throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
572 }
573
574 double edgeLengthSum = 0;
575 int laneNumber = 0;
576 double speedSum = 0;
577 double totalTT = 0;
578 for (MSEdge* edge : myEdges) {
579 edgeLengthSum += edge->getLength();
580 laneNumber += edge->getNumDrivingLanes();
581 speedSum += edge->getSpeedLimit();
582 totalTT += edge->getLength() / edge->getSpeedLimit();
583 }
584 MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
585 for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
586 for (MeanDataValues* meanData : edgeValues) {
587 meanData->addTo(*sumData);
589 meanData->reset();
590 }
591 }
592 }
594 for (int i = 0; i < (int)myEdges.size(); i++) {
595 MSEdge* edge = myEdges[i];
596 std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
598 while (s != nullptr) {
599 for (MeanDataValues* meanData : edgeValues) {
600 s->prepareDetectorForWriting(*meanData);
601 meanData->addTo(*sumData);
603 meanData->reset();
604 }
605 }
606 s = s->getNextSegment();
607 }
608 }
609 }
610
611 if (myDumpEmpty || !sumData->isEmpty()) {
612 writePrefix(dev, *sumData, SUMO_TAG_EDGE, "AGGREGATED");
614 sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)myEdges.size(),
615 myPrintDefaults ? totalTT : -1.);
616 }
617 delete sumData;
618}
619
620
621void
623 if (myTrackVehicles) {
624 throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
625 }
626
627 for (const MSEdge* taz : myTAZ) {
628 double edgeLengthSum = 0;
629 int laneNumber = 0;
630 double speedSum = 0;
631 double totalTT = 0;
632 std::set<const MSEdge*> connected;
633 for (const MSEdge* edge : taz->getSuccessors()) {
634 connected.insert(edge);
635 }
636 for (const MSEdge* edge : taz->getPredecessors()) {
637 connected.insert(edge);
638 }
639 for (const MSEdge* edge : connected) {
640 edgeLengthSum += edge->getLength();
641 laneNumber += edge->getNumDrivingLanes();
642 speedSum += edge->getSpeedLimit();
643 totalTT += edge->getLength() / edge->getSpeedLimit();
644 }
645 MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
646 for (int i = 0; i < (int)myEdges.size(); i++) {
647 MSEdge* edge = myEdges[i];
648 if (connected.count(edge) != 0) {
649 std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
652 while (s != nullptr) {
653 for (MeanDataValues* meanData : edgeValues) {
654 s->prepareDetectorForWriting(*meanData);
655 meanData->addTo(*sumData);
656 }
657 s = s->getNextSegment();
658 }
659 } else {
660 for (MeanDataValues* meanData : edgeValues) {
661 meanData->addTo(*sumData);
662 }
663 }
664 }
665 }
666 if (myDumpEmpty || !sumData->isEmpty()) {
667 writePrefix(dev, *sumData, SUMO_TAG_EDGE, taz->getID());
668 dev.writeAttr(SUMO_ATTR_NUMEDGES, connected.size());
669 sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)connected.size(),
670 myPrintDefaults ? totalTT : -1.);
671 }
672 delete sumData;
673 }
674
676 for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
677 for (MeanDataValues* meanData : edgeValues) {
678 meanData->reset();
679 }
680 }
681 }
682}
683
684
685void
687 const std::vector<MeanDataValues*>& edgeValues,
688 const MSEdge* const edge, SUMOTime startTime, SUMOTime stopTime) {
690 int idx = 0;
691 for (MeanDataValues* const data : edgeValues) {
693 while (s != nullptr) {
694 s->prepareDetectorForWriting(*data, myAmEdgeBased ? -1 : idx);
695 s = s->getNextSegment();
696 }
697 idx++;
698 }
699 if (myAmEdgeBased) {
700 MeanDataValues* const data = edgeValues.front();
701 if (myDumpEmpty || (myPrintModified && edge->getLanes()[0]->isSpeedModified()) || !data->isEmpty()) {
702 writePrefix(dev, *data, SUMO_TAG_EDGE, getEdgeID(edge));
703 data->write(dev, myWrittenAttributes, stopTime - startTime,
704 edge->getNumDrivingLanes(),
705 edge->getSpeedLimit(),
706 myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
707 }
709 data->reset(true);
710 }
711 return;
712 }
713 }
714 if (!myAmEdgeBased) {
715 bool writeCheck = myDumpEmpty;
716 if (!writeCheck) {
717 for (const MeanDataValues* const laneData : edgeValues) {
718 if (!laneData->isEmpty() || (myPrintModified && laneData->getLane()->isSpeedModified())) {
719 writeCheck = true;
720 break;
721 }
722 }
723 }
724 if (writeCheck) {
726 }
727 for (MeanDataValues* const laneData : edgeValues) {
728 const MSLane* const lane = laneData->getLane();
729 if (myDumpEmpty || (myPrintModified && lane->isSpeedModified()) || !laneData->isEmpty()) {
730 writePrefix(dev, *laneData, SUMO_TAG_LANE, lane->getID());
731 laneData->write(dev, myWrittenAttributes, stopTime - startTime, 1, lane->getSpeedLimit(),
732 myPrintDefaults ? lane->getLength() / lane->getSpeedLimit() : -1.);
733 }
735 laneData->reset(true);
736 }
737 }
738 if (writeCheck) {
739 dev.closeTag();
740 }
741 } else {
742 if (myTrackVehicles) {
743 MeanDataValues& meanData = **edgeValues.begin();
744 if (myDumpEmpty || !meanData.isEmpty()) {
745 writePrefix(dev, meanData, SUMO_TAG_EDGE, edge->getID());
746 meanData.write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
747 myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
748 }
750 meanData.reset(true);
751 }
752 } else {
753 MeanDataValues* sumData = createValues(nullptr, edge->getLength(), false);
754 bool writeCheck = myDumpEmpty;
755 for (MeanDataValues* const laneData : edgeValues) {
756 laneData->addTo(*sumData);
757 if (myPrintModified && laneData->getLane()->isSpeedModified()) {
758 writeCheck = true;
759 }
761 laneData->reset();
762 }
763 }
764 if (writeCheck || !sumData->isEmpty()) {
765 writePrefix(dev, *sumData, SUMO_TAG_EDGE, getEdgeID(edge));
766 sumData->write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
767 myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
768 }
769 delete sumData;
770 }
771 }
772}
773
774
775void
780
781
782void
783MSMeanData::writePrefix(OutputDevice& dev, const MeanDataValues& values, const SumoXMLTag tag, const std::string id) const {
784 dev.openTag(tag);
785 dev.writeAttr(SUMO_ATTR_ID, id);
787}
788
789
790void
792 SUMOTime startTime, SUMOTime stopTime) {
793 // check whether this dump shall be written for the current time
794 int numReady = myDumpBegin < stopTime && myDumpEnd - DELTA_T >= startTime ? 1 : 0;
795 if (myTrackVehicles && myDumpBegin < stopTime) {
796 myPendingIntervals.push_back(std::make_pair(startTime, stopTime));
797 numReady = (int)myPendingIntervals.size();
798 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
799 for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
800 numReady = MIN2(numReady, ((MeanDataValueTracker*)*j)->getNumReady());
801 if (numReady == 0) {
802 break;
803 }
804 }
805 if (numReady == 0) {
806 break;
807 }
808 }
809 }
810 const bool partialInterval = startTime < myInitTime;
811 if (numReady == 0 || myTrackVehicles || partialInterval) {
812 resetOnly(stopTime);
813 }
814 if (partialInterval) {
815 return;
816 }
817 while (numReady-- > 0) {
818 if (!myPendingIntervals.empty()) {
819 startTime = myPendingIntervals.front().first;
820 stopTime = myPendingIntervals.front().second;
821 myPendingIntervals.pop_front();
822 }
823 openInterval(dev, startTime, stopTime);
825 writeAggregated(dev, startTime, stopTime);
826 } else if (myAggregate == AggregateType::TAZ) {
827 writeAggregatedTAZ(dev, startTime, stopTime);
828 } else {
829 MSEdgeVector::const_iterator edge = myEdges.begin();
830 for (const std::vector<MeanDataValues*>& measures : myMeasures) {
831 writeEdge(dev, measures, *edge, startTime, stopTime);
832 ++edge;
833 }
834 }
835 dev.closeTag();
836 }
837 dev.flush();
838}
839
840
841void
843 dev.writeXMLHeader("meandata", "meandata_file.xsd");
844}
845
846
847void
849 if (step + DELTA_T == myDumpBegin) {
850 init();
851 }
852}
853
854
855const std::vector<MSMeanData::MeanDataValues*>*
857 auto it = myEdgeIndex.find(edge);
858 if (it != myEdgeIndex.end()) {
859 return &myMeasures[it->second];
860 } else {
861 return nullptr;
862 }
863}
864
865
866const std::vector<MSMoveReminder*>
868 std::vector<MSMoveReminder*> result;
869 for (auto vec : myMeasures) {
870 result.insert(result.end(), vec.begin(), vec.end());
871 }
872 return result;
873}
874
875
876/****************************************************************************/
long long int SUMOTime
Definition GUI.h:36
#define WRITE_ERRORF(...)
Definition MsgHandler.h:296
#define WRITE_ERROR(msg)
Definition MsgHandler.h:295
#define TL(string)
Definition MsgHandler.h:304
SUMOTime DELTA_T
Definition SUMOTime.cpp:38
std::string time2string(SUMOTime t, bool humanReadable)
convert SUMOTime to string (independently of global format setting)
Definition SUMOTime.cpp:91
#define SUMOTime_MAX
Definition SUMOTime.h:34
#define TS
Definition SUMOTime.h:45
#define SIMTIME
Definition SUMOTime.h:65
SumoXMLTag
Numbers representing SUMO-XML - element names.
@ SUMO_TAG_INTERVAL
an aggreagated-output interval
@ SUMO_TAG_LANE
begin/end of the description of a single lane
@ SUMO_TAG_EDGE
begin/end of the description of an edge
std::bitset< 96 > SumoXMLAttrMask
AggregateType
Numbers representing special SUMO-XML-attribute values Information on edgeData/laneData output should...
@ SUMO_ATTR_NUMEDGES
@ SUMO_ATTR_BEGIN
weights: time range begin
@ SUMO_ATTR_END
weights: time range end
@ SUMO_ATTR_SAMPLEDSECONDS
MSMeanData_Net.
@ SUMO_ATTR_ID
T MIN4(T a, T b, T c, T d)
Definition StdDefs.h:107
T MIN2(T a, T b)
Definition StdDefs.h:80
T MAX2(T a, T b)
Definition StdDefs.h:86
MESegment * getSegmentForEdge(const MSEdge &e, double pos=0)
Get the segment for a given edge at a given position.
Definition MELoop.cpp:330
A single mesoscopic segment (cell)
Definition MESegment.h:50
void addDetector(MSMoveReminder *data, int queueIndex=-1)
Adds a data collector for a detector to this segment.
void prepareDetectorForWriting(MSMoveReminder &data, int queueIndex=-1)
Removes a data collector for a detector from this segment.
MESegment * getNextSegment() const
Returns the following segment on the same edge (0 if it is the last).
Definition MESegment.h:238
static double speedAfterTime(const double t, const double oldSpeed, const double dist)
Calculates the speed after a time t \in [0,TS] given the initial speed and the distance traveled in a...
static double passingTime(const double lastPos, const double passedPos, const double currentPos, const double lastSpeed, const double currentSpeed)
Calculates the time at which the position passedPosition has been passed In case of a ballistic updat...
Base of value-generating classes (detectors)
const MSEdgeVector & getEdges() const
Returns loaded edges.
A road/street connecting two junctions.
Definition MSEdge.h:77
static const MSEdgeVector & getAllEdges()
Returns all edges with a numerical id.
Definition MSEdge.cpp:1114
const std::vector< MSLane * > & getLanes() const
Returns this edge's lanes.
Definition MSEdge.h:168
int getNumDrivingLanes() const
return the number of lanes that permit non-weak modes if the edge allows non weak modes and the numbe...
Definition MSEdge.cpp:1542
double getSpeedLimit() const
Returns the speed limit of the edge @caution The speed limit of the first lane is retured; should pro...
Definition MSEdge.cpp:1186
double getLength() const
return the length of the edge
Definition MSEdge.h:694
static bool gUseMesoSim
Definition MSGlobals.h:106
static MELoop * gMesoNet
mesoscopic simulation infrastructure
Definition MSGlobals.h:112
static bool gSemiImplicitEulerUpdate
Definition MSGlobals.h:53
static int gNumSimThreads
how many threads to use for simulation
Definition MSGlobals.h:146
Representation of a lane in the micro simulation.
Definition MSLane.h:84
double getSpeedLimit() const
Returns the lane's maximum allowed speed.
Definition MSLane.h:603
double getLength() const
Returns the lane's length.
Definition MSLane.h:617
bool isSpeedModified() const
Definition MSLane.h:595
Definition MSMeanData.h:276
Data structure for mean (aggregated) edge/lane values for tracked vehicles.
Definition MSMeanData.h:202
double getSamples() const
Returns the number of collected sample seconds.
MeanDataValueTracker(MSLane *const lane, const double length, const MSMeanData *const parent)
Constructor.
bool notifyLeave(SUMOTrafficObject &veh, double lastPos, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Called if the vehicle leaves the reminder's lane.
void addTo(MSMeanData::MeanDataValues &val) const
Add the values of this to the given one and store them there.
void 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)
Internal notification about the vehicle moves.
void write(OutputDevice &dev, const SumoXMLAttrMask &attributeMask, const SUMOTime period, const int numLanes, const double speedLimit, const double defaultTravelTime, const int numVehicles=-1) const
Writes output values into the given stream.
virtual ~MeanDataValueTracker()
Destructor.
bool notifyEnter(SUMOTrafficObject &veh, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Computes current values and adds them to their sums.
std::list< TrackerEntry * > myCurrentData
The currently active meandata "intervals".
Definition MSMeanData.h:301
bool isEmpty() const
Returns whether any data was collected.
void reset(bool afterWrite)
Resets values so they may be used for the next interval.
Data structure for mean (aggregated) edge/lane values.
Definition MSMeanData.h:66
virtual void addTo(MeanDataValues &val) const =0
Add the values of this to the given one and store them there.
virtual bool notifyLeave(SUMOTrafficObject &veh, double lastPos, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Called if the vehicle leaves the reminder's lane.
MeanDataValues(MSLane *const lane, const double length, const bool doAdd, const MSMeanData *const parent)
Constructor.
virtual void write(OutputDevice &dev, const SumoXMLAttrMask &attributeMask, const SUMOTime period, const int numLanes, const double speedLimit, const double defaultTravelTime, const int numVehicles=-1) const =0
Writes output values into the given stream.
virtual void update()
Called if a per timestep update is needed. Default does nothing.
virtual bool isEmpty() const
Returns whether any data was collected.
bool notifyMove(SUMOTrafficObject &veh, double oldPos, double newPos, double newSpeed)
Checks whether the reminder still has to be notified about the vehicle moves.
virtual ~MeanDataValues()
Destructor.
virtual void reset(bool afterWrite=false)=0
Resets values so they may be used for the next interval.
virtual bool notifyEnter(SUMOTrafficObject &veh, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Called if the vehicle enters the reminder's lane.
virtual double getSamples() const
Returns the number of collected sample seconds.
Data collector for edges/lanes.
Definition MSMeanData.h:57
virtual MSMeanData::MeanDataValues * createValues(MSLane *const lane, const double length, const bool doAdd) const =0
Create an instance of MeanDataValues.
void writeXMLOutput(OutputDevice &dev, SUMOTime startTime, SUMOTime stopTime)
Writes collected values into the given stream.
virtual void writePrefix(OutputDevice &dev, const MeanDataValues &values, const SumoXMLTag tag, const std::string id) const
Writes the surrounding element into the given stream.
const bool myDumpInternal
Whether internal lanes/edges shall be written.
Definition MSMeanData.h:525
bool myPrintDefaults
Whether empty lanes/edges shall be written with default values.
Definition MSMeanData.h:519
const AggregateType myAggregate
whether the data for all edges shall be aggregated
Definition MSMeanData.h:534
const SUMOTime myDumpBegin
The first and the last time step to write information (-1 indicates always)
Definition MSMeanData.h:505
virtual void detectorUpdate(const SUMOTime step)
Updates the detector.
virtual ~MSMeanData()
Destructor.
MSMeanData(const std::string &id, const SUMOTime dumpBegin, const SUMOTime dumpEnd, const bool useLanes, const std::string &excludeEmpty, const bool withInternal, const bool trackVehicles, const int detectPersons, const double minSamples, const double maxTravelTime, const std::string &vTypes, const std::string &writeAttributes, const std::vector< MSEdge * > &edges, AggregateType aggregate)
Constructor.
void writeAggregated(OutputDevice &dev, SUMOTime startTime, SUMOTime stopTime)
Writes aggregated data of all edge values into the given stream.
void init()
Adds the value collectors to all relevant edges.
const double myMinSamples
the minimum sample seconds
Definition MSMeanData.h:489
void resetOnly(SUMOTime stopTime)
Resets network value in order to allow processing of the next interval.
void writeAggregatedTAZ(OutputDevice &dev, SUMOTime startTime, SUMOTime stopTime)
Writes aggregated data for each TAZ into the given stream.
std::map< const MSEdge *, int > myEdgeIndex
The index in myEdges / myMeasures.
Definition MSMeanData.h:516
const SumoXMLAttrMask myWrittenAttributes
bit mask for checking attributes to be written
Definition MSMeanData.h:531
const bool myAmEdgeBased
Information whether the output shall be edge-based (not lane-based)
Definition MSMeanData.h:501
SUMOTime myInitTime
time at which init was called();
Definition MSMeanData.h:508
const double myMaxTravelTime
the maximum travel time to write
Definition MSMeanData.h:492
bool myDumpEmpty
Whether empty lanes/edges shall be written.
Definition MSMeanData.h:498
const SUMOTime myDumpEnd
Definition MSMeanData.h:505
std::list< std::pair< SUMOTime, SUMOTime > > myPendingIntervals
The intervals for which output still has to be generated (only in the tracking case)
Definition MSMeanData.h:537
const std::vector< MeanDataValues * > * getEdgeValues(const MSEdge *edge) const
void writeEdge(OutputDevice &dev, const std::vector< MeanDataValues * > &edgeValues, const MSEdge *const edge, SUMOTime startTime, SUMOTime stopTime)
Writes edge values into the given stream.
virtual std::string getEdgeID(const MSEdge *const edge)
Return the relevant edge id.
virtual void writeXMLDetectorProlog(OutputDevice &dev) const
Opens the XML-output using "netstats" as root element.
MSEdgeVector myEdges
The corresponding first edges.
Definition MSMeanData.h:511
const std::vector< MSMoveReminder * > getReminders() const
retrieve all MeanDataValues
virtual void openInterval(OutputDevice &dev, const SUMOTime startTime, const SUMOTime stopTime)
Writes the interval opener.
bool myPrintModified
Whether only empty lanes/edges which have been modified shall be written.
Definition MSMeanData.h:522
std::vector< std::vector< MeanDataValues * > > myMeasures
Value collectors; sorted by edge, then by lane.
Definition MSMeanData.h:495
const bool myTrackVehicles
Whether vehicles are tracked.
Definition MSMeanData.h:528
ConstMSEdgeVector myTAZ
Definition MSMeanData.h:513
Something on a lane to be noticed about vehicle movement.
void setDescription(const std::string &description)
Notification
Definition of a vehicle state.
@ NOTIFICATION_SEGMENT
The vehicle changes the segment (meso only)
@ NOTIFICATION_JUNCTION
The vehicle arrived at a junction.
static MSNet * getInstance()
Returns the pointer to the unique instance of MSNet (singleton).
Definition MSNet.cpp:187
SUMOTime getCurrentTimeStep() const
Returns the current simulation step.
Definition MSNet.h:334
virtual bool skipFinalReset() const
gui may prevent final meanData reset to keep live data visible
Definition MSNet.h:839
MSEdgeControl & getEdgeControl()
Returns the edge control.
Definition MSNet.h:435
double getLength() const
Get vehicle's length [m].
std::string myID
The name of the object.
Definition Named.h:125
void addTo(const StoringVisitor &cont) const
Adds this object to the given container.
Definition Named.h:118
const std::string & getID() const
Returns the id.
Definition Named.h:74
Static storage of an output device and its base (abstract) implementation.
OutputDevice & writeAttr(const SumoXMLAttr attr, const T &val)
writes a named attribute
OutputDevice & openTag(const std::string &xmlElement)
Opens an XML tag.
bool closeTag(const std::string &comment="")
Closes the most recently opened tag and optionally adds a comment.
bool writeXMLHeader(const std::string &rootElement, const std::string &schemaFile, std::map< SumoXMLAttr, std::string > attrs=std::map< SumoXMLAttr, std::string >(), bool includeConfig=true)
Writes an XML header with optional configuration.
OutputDevice & writeOptionalAttr(const SumoXMLAttr attr, const T &val, const SumoXMLAttrMask &attributeMask, const bool isNull=false)
writes a named attribute unless filtered
Representation of a vehicle, person, or container.
virtual const MSVehicleType & getVehicleType() const =0
Returns the object's "vehicle" type.
virtual double getPreviousSpeed() const =0
Returns the object's previous speed.
virtual bool hasArrived() const =0
Returns whether this object has arrived.
A scoped lock which only triggers on condition.
static bool toBool(const std::string &sData)
converts a string into the bool value described by it by calling the char-type converter
#define UNUSED_PARAMETER(x)