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(std::make_shared<TrackerEntry>(parent->createValues(lane, length, false)));
293}
294
295
297
298
299void
301 if (afterWrite) {
302 if (!myCurrentData.empty()) {
303 myCurrentData.pop_front();
304 }
305 } else {
306 myCurrentData.push_back(std::make_shared<TrackerEntry>(myParent->createValues(myLane, myLaneLength, false)));
307 }
308}
309
310
311void
313 myCurrentData.front()->myValues->addTo(val);
314}
315
316
317void
318MSMeanData::MeanDataValueTracker::notifyMoveInternal(const SUMOTrafficObject& veh, const double frontOnLane, const double timeOnLane, const double meanSpeedFrontOnLane, const double meanSpeedVehicleOnLane, const double travelledDistanceFrontOnLane, const double travelledDistanceVehicleOnLane, const double meanLengthOnLane) {
319 myTrackedData[&veh]->myValues->notifyMoveInternal(veh, frontOnLane, timeOnLane, meanSpeedFrontOnLane, meanSpeedVehicleOnLane, travelledDistanceFrontOnLane, travelledDistanceVehicleOnLane, meanLengthOnLane);
320}
321
322
323bool
325 if (myParent == nullptr || reason != MSMoveReminder::NOTIFICATION_SEGMENT) {
326 myTrackedData[&veh]->myNumVehicleLeft++;
327 }
328 return myTrackedData[&veh]->myValues->notifyLeave(veh, lastPos, reason);
329}
330
331
332bool
334#ifdef DEBUG_NOTIFY_ENTER
335 std::cout << "\n" << SIMTIME << " MSMeanData::MeanDataValueTracker: veh '" << veh.getID() << "' enters lane '" << enteredLane->getID() << "'" << std::endl;
336#else
337 UNUSED_PARAMETER(enteredLane);
338#endif
340 return true;
341 }
342 if (myParent->vehicleApplies(veh) && myTrackedData.find(&veh) == myTrackedData.end()) {
343 myTrackedData[&veh] = myCurrentData.back();
344 myTrackedData[&veh]->myNumVehicleEntered++;
345 if (!myTrackedData[&veh]->myValues->notifyEnter(veh, reason)) {
346 myTrackedData[&veh]->myNumVehicleLeft++;
347 myTrackedData.erase(&veh);
348 return false;
349 }
350 return true;
351 }
352 return false;
353}
354
355
356bool
358 return myCurrentData.front()->myValues->isEmpty();
359}
360
361
362void
364 const SumoXMLAttrMask& attributeMask,
365 const SUMOTime period,
366 const int numLanes,
367 const double speedLimit,
368 const double defaultTravelTime,
369 const int /*numVehicles*/) const {
370 myCurrentData.front()->myValues->write(dev, attributeMask, period, numLanes, speedLimit,
371 defaultTravelTime,
372 myCurrentData.front()->myNumVehicleEntered);
373}
374
375
376int
378 int result = 0;
379 for (const auto& it : myCurrentData) {
380 if (it->myNumVehicleEntered == it->myNumVehicleLeft) {
381 result++;
382 } else {
383 break;
384 }
385 }
386 return result;
387}
388
389
390double
392 return myCurrentData.front()->myValues->getSamples();
393}
394
395
396// ---------------------------------------------------------------------------
397// MSMeanData - methods
398// ---------------------------------------------------------------------------
399MSMeanData::MSMeanData(const std::string& id,
400 const SUMOTime dumpBegin, const SUMOTime dumpEnd,
401 const bool useLanes, const std::string& excludeEmpty, const bool withInternal,
402 const bool trackVehicles,
403 const int detectPersons,
404 const double maxTravelTime,
405 const double minSamples,
406 const std::string& vTypes,
407 const std::string& writeAttributes,
408 const std::vector<MSEdge*>& edges,
409 AggregateType aggregate) :
410 MSDetectorFileOutput(id, vTypes, "", detectPersons),
411 myMinSamples(minSamples),
412 myMaxTravelTime(maxTravelTime),
413 myAmEdgeBased(!useLanes),
414 myDumpBegin(dumpBegin),
415 myDumpEnd(dumpEnd),
417 myEdges(edges),
418 myDumpInternal(withInternal && MSGlobals::gUsingInternalLanes),
419 myTrackVehicles(trackVehicles),
420 myWrittenAttributes(OutputDevice::parseWrittenAttributes(StringTokenizer(writeAttributes).getVector(), "meandata '" + id + "'")),
421 myAggregate(aggregate) {
422 try {
423 myDumpEmpty = !StringUtils::toBool(excludeEmpty);
424 } catch (const BoolFormatException&) {
425 if (excludeEmpty == "default" || excludeEmpty == "defaults") {
426 myPrintDefaults = true;
427 } else if (excludeEmpty == "modified") {
428 myDumpEmpty = false;
429 myPrintDefaults = true;
430 myPrintModified = true;
431 } else {
432 throw;
433 }
434 }
435}
436
437
438void
441 if (myEdges.empty()) {
442 // use all edges by default
443 for (MSEdge* const edge : MSNet::getInstance()->getEdgeControl().getEdges()) {
444 if ((myDumpInternal || !edge->isInternal()) &&
445 ((detectsPersons() && myDumpInternal) || (!edge->isCrossing() && !edge->isWalkingArea()))) {
446 myEdges.push_back(edge);
447 }
448 }
449 }
450 int index = 0;
451 for (MSEdge* edge : myEdges) {
452 myMeasures.push_back(std::vector<MeanDataValues*>());
453 myEdgeIndex[edge] = index++;
454 const std::vector<MSLane*>& lanes = edge->getLanes();
456 MeanDataValues* data;
457 if (!myAmEdgeBased) {
458 for (MSLane* const lane : lanes) {
459 data = createValues(lane, lanes[0]->getLength(), false);
460 myMeasures.back().push_back(data);
462 while (s != nullptr) {
463 s->addDetector(data, lane->getIndex());
464 s->prepareDetectorForWriting(*data, lane->getIndex());
465 s = s->getNextSegment();
466 }
467 data->reset();
468 data->reset(true);
469 }
470 } else {
471 if (myTrackVehicles) {
472 data = new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this);
473 } else {
474 data = createValues(nullptr, lanes[0]->getLength(), false);
475 }
476 data->setDescription("meandata_" + getID() + "|" + edge->getID());
477 myMeasures.back().push_back(data);
479 while (s != nullptr) {
480 s->addDetector(data);
482 s = s->getNextSegment();
483 }
484 data->reset();
485 data->reset(true);
486 }
487 continue;
488 }
490 myMeasures.back().push_back(new MeanDataValueTracker(nullptr, lanes[0]->getLength(), this));
491 }
492 for (MSLane* const lane : lanes) {
493 if (myTrackVehicles) {
494 if (myAmEdgeBased) {
495 lane->addMoveReminder(myMeasures.back().back());
496 } else {
497 myMeasures.back().push_back(new MeanDataValueTracker(lane, lane->getLength(), this));
498 }
499 } else {
500 myMeasures.back().push_back(createValues(lane, lane->getLength(), true));
501 }
502 }
503 }
505 for (const MSEdge* e : MSEdge::getAllEdges()) {
506 if (e->isTazConnector()) {
507 myTAZ.push_back(e);
508 }
509 }
510 }
511}
512
513
515 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
516 for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
517 delete *j;
518 }
519 }
520}
521
522
523void
525 UNUSED_PARAMETER(stopTime);
527 MSEdgeVector::iterator edge = myEdges.begin();
528 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i, ++edge) {
530 for (MeanDataValues* data : *i) {
531 while (s != nullptr) {
533 s = s->getNextSegment();
534 }
535 data->reset();
536 }
537 }
538 return;
539 }
540 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
541 for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
542 (*j)->reset();
543 }
544 }
545}
546
547
548std::string
549MSMeanData::getEdgeID(const MSEdge* const edge) {
550 return edge->getID();
551}
552
553
554void
556 if (myTrackVehicles) {
557 throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
558 }
559
560 double edgeLengthSum = 0;
561 int laneNumber = 0;
562 double speedSum = 0;
563 double totalTT = 0;
564 for (MSEdge* edge : myEdges) {
565 edgeLengthSum += edge->getLength();
566 laneNumber += edge->getNumDrivingLanes();
567 speedSum += edge->getSpeedLimit();
568 totalTT += edge->getLength() / edge->getSpeedLimit();
569 }
570 MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
571 for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
572 for (MeanDataValues* meanData : edgeValues) {
573 meanData->addTo(*sumData);
575 meanData->reset();
576 }
577 }
578 }
580 for (int i = 0; i < (int)myEdges.size(); i++) {
581 MSEdge* edge = myEdges[i];
582 std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
584 while (s != nullptr) {
585 for (MeanDataValues* meanData : edgeValues) {
586 s->prepareDetectorForWriting(*meanData);
587 meanData->addTo(*sumData);
589 meanData->reset();
590 }
591 }
592 s = s->getNextSegment();
593 }
594 }
595 }
596
597 if (myDumpEmpty || !sumData->isEmpty()) {
598 writePrefix(dev, *sumData, SUMO_TAG_EDGE, "AGGREGATED");
600 sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)myEdges.size(),
601 myPrintDefaults ? totalTT : -1.);
602 }
603 delete sumData;
604}
605
606
607void
609 if (myTrackVehicles) {
610 throw ProcessError(TL("aggregated meanData output not yet implemented for trackVehicles"));
611 }
612
613 for (const MSEdge* taz : myTAZ) {
614 double edgeLengthSum = 0;
615 int laneNumber = 0;
616 double speedSum = 0;
617 double totalTT = 0;
618 std::set<const MSEdge*> connected;
619 for (const MSEdge* edge : taz->getSuccessors()) {
620 connected.insert(edge);
621 }
622 for (const MSEdge* edge : taz->getPredecessors()) {
623 connected.insert(edge);
624 }
625 for (const MSEdge* edge : connected) {
626 edgeLengthSum += edge->getLength();
627 laneNumber += edge->getNumDrivingLanes();
628 speedSum += edge->getSpeedLimit();
629 totalTT += edge->getLength() / edge->getSpeedLimit();
630 }
631 MeanDataValues* sumData = createValues(nullptr, edgeLengthSum, false);
632 for (int i = 0; i < (int)myEdges.size(); i++) {
633 MSEdge* edge = myEdges[i];
634 if (connected.count(edge) != 0) {
635 std::vector<MeanDataValues*>& edgeValues = myMeasures[i];
638 while (s != nullptr) {
639 for (MeanDataValues* meanData : edgeValues) {
640 s->prepareDetectorForWriting(*meanData);
641 meanData->addTo(*sumData);
642 }
643 s = s->getNextSegment();
644 }
645 } else {
646 for (MeanDataValues* meanData : edgeValues) {
647 meanData->addTo(*sumData);
648 }
649 }
650 }
651 }
652 if (myDumpEmpty || !sumData->isEmpty()) {
653 writePrefix(dev, *sumData, SUMO_TAG_EDGE, taz->getID());
654 dev.writeAttr(SUMO_ATTR_NUMEDGES, connected.size());
655 sumData->write(dev, myWrittenAttributes, stopTime - startTime, laneNumber, speedSum / (double)connected.size(),
656 myPrintDefaults ? totalTT : -1.);
657 }
658 delete sumData;
659 }
660
662 for (const std::vector<MeanDataValues*>& edgeValues : myMeasures) {
663 for (MeanDataValues* meanData : edgeValues) {
664 meanData->reset();
665 }
666 }
667 }
668}
669
670
671void
673 const std::vector<MeanDataValues*>& edgeValues,
674 const MSEdge* const edge, SUMOTime startTime, SUMOTime stopTime) {
676 int idx = 0;
677 for (MeanDataValues* const data : edgeValues) {
679 while (s != nullptr) {
680 s->prepareDetectorForWriting(*data, myAmEdgeBased ? -1 : idx);
681 s = s->getNextSegment();
682 }
683 idx++;
684 }
685 if (myAmEdgeBased) {
686 MeanDataValues* const data = edgeValues.front();
687 if (myDumpEmpty || (myPrintModified && edge->getLanes()[0]->isSpeedModified()) || !data->isEmpty()) {
688 writePrefix(dev, *data, SUMO_TAG_EDGE, getEdgeID(edge));
689 data->write(dev, myWrittenAttributes, stopTime - startTime,
690 edge->getNumDrivingLanes(),
691 edge->getSpeedLimit(),
692 myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
693 }
695 data->reset(true);
696 }
697 return;
698 }
699 }
700 if (!myAmEdgeBased) {
701 bool writeCheck = myDumpEmpty;
702 if (!writeCheck) {
703 for (const MeanDataValues* const laneData : edgeValues) {
704 if (!laneData->isEmpty() || (myPrintModified && laneData->getLane()->isSpeedModified())) {
705 writeCheck = true;
706 break;
707 }
708 }
709 }
710 if (writeCheck) {
712 }
713 for (MeanDataValues* const laneData : edgeValues) {
714 const MSLane* const lane = laneData->getLane();
715 if (myDumpEmpty || (myPrintModified && lane->isSpeedModified()) || !laneData->isEmpty()) {
716 writePrefix(dev, *laneData, SUMO_TAG_LANE, lane->getID());
717 laneData->write(dev, myWrittenAttributes, stopTime - startTime, 1, lane->getSpeedLimit(),
718 myPrintDefaults ? lane->getLength() / lane->getSpeedLimit() : -1.);
719 }
721 laneData->reset(true);
722 }
723 }
724 if (writeCheck) {
725 dev.closeTag();
726 }
727 } else {
728 if (myTrackVehicles) {
729 MeanDataValues& meanData = **edgeValues.begin();
730 if (myDumpEmpty || !meanData.isEmpty()) {
731 writePrefix(dev, meanData, SUMO_TAG_EDGE, edge->getID());
732 meanData.write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
733 myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
734 }
736 meanData.reset(true);
737 }
738 } else {
739 MeanDataValues* sumData = createValues(nullptr, edge->getLength(), false);
740 bool writeCheck = myDumpEmpty;
741 for (MeanDataValues* const laneData : edgeValues) {
742 laneData->addTo(*sumData);
743 if (myPrintModified && laneData->getLane()->isSpeedModified()) {
744 writeCheck = true;
745 }
747 laneData->reset();
748 }
749 }
750 if (writeCheck || !sumData->isEmpty()) {
751 writePrefix(dev, *sumData, SUMO_TAG_EDGE, getEdgeID(edge));
752 sumData->write(dev, myWrittenAttributes, stopTime - startTime, edge->getNumDrivingLanes(), edge->getSpeedLimit(),
753 myPrintDefaults ? edge->getLength() / edge->getSpeedLimit() : -1.);
754 }
755 delete sumData;
756 }
757 }
758}
759
760
761void
766
767
768void
769MSMeanData::writePrefix(OutputDevice& dev, const MeanDataValues& values, const SumoXMLTag tag, const std::string id) const {
770 dev.openTag(tag);
771 dev.writeAttr(SUMO_ATTR_ID, id);
773}
774
775
776void
778 SUMOTime startTime, SUMOTime stopTime) {
779 // check whether this dump shall be written for the current time
780 int numReady = myDumpBegin < stopTime && myDumpEnd - DELTA_T >= startTime ? 1 : 0;
781 if (myTrackVehicles && myDumpBegin < stopTime) {
782 myPendingIntervals.push_back(std::make_pair(startTime, stopTime));
783 numReady = (int)myPendingIntervals.size();
784 for (std::vector<std::vector<MeanDataValues*> >::const_iterator i = myMeasures.begin(); i != myMeasures.end(); ++i) {
785 for (std::vector<MeanDataValues*>::const_iterator j = (*i).begin(); j != (*i).end(); ++j) {
786 numReady = MIN2(numReady, ((MeanDataValueTracker*)*j)->getNumReady());
787 if (numReady == 0) {
788 break;
789 }
790 }
791 if (numReady == 0) {
792 break;
793 }
794 }
795 }
796 const bool partialInterval = startTime < myInitTime;
797 if (numReady == 0 || myTrackVehicles || partialInterval) {
798 resetOnly(stopTime);
799 }
800 if (partialInterval) {
801 return;
802 }
803 while (numReady-- > 0) {
804 if (!myPendingIntervals.empty()) {
805 startTime = myPendingIntervals.front().first;
806 stopTime = myPendingIntervals.front().second;
807 myPendingIntervals.pop_front();
808 }
809 openInterval(dev, startTime, stopTime);
811 writeAggregated(dev, startTime, stopTime);
812 } else if (myAggregate == AggregateType::TAZ) {
813 writeAggregatedTAZ(dev, startTime, stopTime);
814 } else {
815 MSEdgeVector::const_iterator edge = myEdges.begin();
816 for (const std::vector<MeanDataValues*>& measures : myMeasures) {
817 writeEdge(dev, measures, *edge, startTime, stopTime);
818 ++edge;
819 }
820 }
821 dev.closeTag();
822 }
823 dev.flush();
824}
825
826
827void
829 dev.writeXMLHeader("meandata", "meandata_file.xsd");
830}
831
832
833void
835 if (step + DELTA_T == myDumpBegin) {
836 init();
837 }
838}
839
840
841const std::vector<MSMeanData::MeanDataValues*>*
843 auto it = myEdgeIndex.find(edge);
844 if (it != myEdgeIndex.end()) {
845 return &myMeasures[it->second];
846 } else {
847 return nullptr;
848 }
849}
850
851
852const std::vector<MSMoveReminder*>
854 std::vector<MSMoveReminder*> result;
855 for (auto vec : myMeasures) {
856 result.insert(result.end(), vec.begin(), vec.end());
857 }
858 return result;
859}
860
861
862/****************************************************************************/
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
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.
std::list< std::shared_ptr< TrackerEntry > > myCurrentData
The currently active meandata "intervals".
Definition MSMeanData.h:301
virtual ~MeanDataValueTracker()
Destructor.
bool notifyEnter(SUMOTrafficObject &veh, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Computes current values and adds them to their sums.
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 & openTag(const std::string &xmlElement)
Opens an XML tag.
OutputDevice & writeAttr(const ATTR_TYPE &attr, const T &val, const bool isNull=false)
writes a named attribute
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)