Eclipse SUMO - Simulation of Urban MObility
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
MSTriggeredRerouter.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-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/****************************************************************************/
22// Reroutes vehicles passing an edge
23/****************************************************************************/
24#include <config.h>
25
26#include <string>
27#include <algorithm>
28#ifdef HAVE_FOX
30#endif
43#include <microsim/MSLane.h>
44#include <microsim/MSLink.h>
45#include <microsim/MSVehicle.h>
46#include <microsim/MSRoute.h>
47#include <microsim/MSEdge.h>
49#include <microsim/MSNet.h>
51#include <microsim/MSGlobals.h>
53#include <microsim/MSStop.h>
59#include "MSTriggeredRerouter.h"
60
61#include <mesosim/MELoop.h>
62#include <mesosim/MESegment.h>
63
64//#define DEBUG_REROUTER
65#define DEBUGCOND (veh.isSelected())
66//#define DEBUGCOND (true)
67//#define DEBUGCOND (veh.getID() == "")
68
69// ===========================================================================
70// static member definition
71// ===========================================================================
72MSEdge MSTriggeredRerouter::mySpecialDest_keepDestination("MSTriggeredRerouter_keepDestination", -1, SumoXMLEdgeFunc::UNKNOWN, "", "", -1, 0);
73MSEdge MSTriggeredRerouter::mySpecialDest_terminateRoute("MSTriggeredRerouter_terminateRoute", -1, SumoXMLEdgeFunc::UNKNOWN, "", "", -1, 0);
74std::map<std::string, MSTriggeredRerouter*> MSTriggeredRerouter::myInstances;
75
76
77// ===========================================================================
78// method definitions
79// ===========================================================================
81 const MSEdgeVector& edges, double prob, bool off, bool optional,
82 SUMOTime timeThreshold, const std::string& vTypes, const Position& pos, const double radius) :
83 Named(id),
86 myEdges(edges),
87 myProbability(prob),
88 myUserProbability(prob),
89 myAmInUserMode(false),
90 myAmOptional(optional),
91 myPosition(pos),
92 myRadius(radius),
93 myTimeThreshold(timeThreshold),
94 myHaveParkProbs(false) {
95 myInstances[id] = this;
96 // build actors
97 for (const MSEdge* const e : edges) {
100 }
101 for (MSLane* const lane : e->getLanes()) {
102 lane->addMoveReminder(this);
103 }
104 }
105 if (off) {
106 setUserMode(true);
108 }
109 const std::vector<std::string> vt = StringTokenizer(vTypes).getVector();
110 myVehicleTypes.insert(vt.begin(), vt.end());
112 myPosition = edges.front()->getLanes()[0]->getShape()[0];
113 }
114}
115
116
120
121
122// ------------ loading begin
123void
125 const SUMOSAXAttributes& attrs) {
126 if (element == SUMO_TAG_INTERVAL) {
127 bool ok = true;
132 throw ProcessError(TLF("rerouter '%': interval end % is not after begin %.", getID(),
135 }
136 }
137 if (element == SUMO_TAG_DEST_PROB_REROUTE) {
138 // by giving probabilities of new destinations
139 // get the destination edge
140 std::string dest = attrs.getStringSecure(SUMO_ATTR_ID, "");
141 if (dest == "") {
142 throw ProcessError(TLF("rerouter '%': destProbReroute has no destination edge id.", getID()));
143 }
144 MSEdge* to = MSEdge::dictionary(dest);
145 if (to == nullptr) {
146 if (dest == "keepDestination") {
148 } else if (dest == "terminateRoute") {
150 } else {
151 throw ProcessError(TLF("rerouter '%': Destination edge '%' is not known.", getID(), dest));
152 }
153 }
154 // get the probability to reroute
155 bool ok = true;
156 double prob = attrs.getOpt<double>(SUMO_ATTR_PROB, getID().c_str(), ok, 1.);
157 if (!ok) {
158 throw ProcessError();
159 }
160 if (prob < 0) {
161 throw ProcessError(TLF("rerouter '%': Attribute 'probability' for destination '%' is negative (must not).", getID(), dest));
162 }
163 // add
165 }
166
167 if (element == SUMO_TAG_CLOSING_REROUTE) {
168 // by closing edge
169 const std::string& closed_id = attrs.getStringSecure(SUMO_ATTR_ID, "");
170 MSEdge* const closed = MSEdge::dictionary(closed_id);
171 if (closed == nullptr) {
172 throw ProcessError(TLF("rerouter '%': Edge '%' to close is not known.", getID(), closed_id));
173 }
174 myParsedRerouteInterval.closed.push_back(closed);
175 bool ok;
176 const std::string allow = attrs.getOpt<std::string>(SUMO_ATTR_ALLOW, getID().c_str(), ok, "", false);
177 const std::string disallow = attrs.getOpt<std::string>(SUMO_ATTR_DISALLOW, getID().c_str(), ok, "");
179 }
180
181 if (element == SUMO_TAG_CLOSING_LANE_REROUTE) {
182 // by closing lane
183 std::string closed_id = attrs.getStringSecure(SUMO_ATTR_ID, "");
184 MSLane* closed = MSLane::dictionary(closed_id);
185 if (closed == nullptr) {
186 throw ProcessError(TLF("rerouter '%': Lane '%' to close is not known.", getID(), closed_id));
187 }
188 myParsedRerouteInterval.closedLanes.push_back(closed);
189 bool ok;
191 const std::string allow = attrs.getOpt<std::string>(SUMO_ATTR_ALLOW, getID().c_str(), ok, "", false);
192 const std::string disallow = attrs.getOpt<std::string>(SUMO_ATTR_DISALLOW, getID().c_str(), ok, "");
194 } else {
195 // lane closing only makes sense if the lane really receives reduced permissions
197 }
198 }
199
200 if (element == SUMO_TAG_ROUTE_PROB_REROUTE) {
201 // by explicit rerouting using routes
202 // check if route exists
203 std::string routeStr = attrs.getStringSecure(SUMO_ATTR_ID, "");
204 if (routeStr == "") {
205 throw ProcessError(TLF("rerouter '%': routeProbReroute has no alternative route id.", getID()));
206 }
207 ConstMSRoutePtr route = MSRoute::dictionary(routeStr);
208 if (route == nullptr) {
209 throw ProcessError(TLF("rerouter '%': Alternative route '%' does not exist.", getID(), routeStr));
210 }
211
212 // get the probability to reroute
213 bool ok = true;
214 double prob = attrs.getOpt<double>(SUMO_ATTR_PROB, getID().c_str(), ok, 1.);
215 if (!ok) {
216 throw ProcessError();
217 }
218 if (prob < 0) {
219 throw ProcessError(TLF("rerouter '%': Attribute 'probability' for alternative route '%' is negative (must not).", getID(), routeStr));
220 }
221 // add
223 }
224
225 if (element == SUMO_TAG_PARKING_AREA_REROUTE) {
226 std::string parkingarea = attrs.getStringSecure(SUMO_ATTR_ID, "");
227 if (parkingarea == "") {
228 throw ProcessError(TLF("rerouter '%': parkingAreaReroute requires a parkingArea id.", getID()));
229 }
231 if (pa == nullptr) {
232 throw ProcessError(TLF("rerouter '%': parkingArea '%' is not known.", getID(), parkingarea));
233 }
234 // get the probability to reroute
235 bool ok = true;
236 const double prob = attrs.getOpt<double>(SUMO_ATTR_PROB, getID().c_str(), ok, 1.);
237 if (!ok) {
238 throw ProcessError();
239 }
240 if (prob < 0) {
241 throw ProcessError(TLF("rerouter '%': Attribute 'probability' for parkingArea '%' is negative (must not).", getID(), parkingarea));
242 }
243 const bool visible = attrs.getOpt<bool>(SUMO_ATTR_VISIBLE, getID().c_str(), ok, false);
244 // add
245 myParsedRerouteInterval.parkProbs.add(std::make_pair(pa, visible), prob);
246 myHaveParkProbs = true;
247 }
248
249 if (element == SUMO_TAG_VIA_PROB_REROUTE) {
250 // by giving probabilities of vias
251 std::string viaID = attrs.getStringSecure(SUMO_ATTR_ID, "");
252 if (viaID == "") {
253 throw ProcessError(TLF("rerouter '%': No via edge id given.", getID()));
254 }
255 MSEdge* const via = MSEdge::dictionary(viaID);
256 if (via == nullptr) {
257 throw ProcessError(TLF("rerouter '%': Via Edge '%' is not known.", getID(), viaID));
258 }
259 // get the probability to reroute
260 bool ok = true;
261 double prob = attrs.getOpt<double>(SUMO_ATTR_PROB, getID().c_str(), ok, 1.);
262 if (!ok) {
263 throw ProcessError();
264 }
265 if (prob < 0) {
266 throw ProcessError(TLF("rerouter '%': Attribute 'probability' for via '%' is negative (must not).", getID(), viaID));
267 }
268 // add
271 }
272 if (element == SUMO_TAG_OVERTAKING_REROUTE) {
273 // for letting a slow train use a siding to be overtaken by a fast train
274 bool ok = true;
275 for (const std::string& edgeID : attrs.get<std::vector<std::string> >(SUMO_ATTR_MAIN, getID().c_str(), ok)) {
276 MSEdge* edge = MSEdge::dictionary(edgeID);
277 if (edge == nullptr) {
278 throw InvalidArgument("The main edge '" + edgeID + "' to use within rerouter '" + getID() + "' is not known.");
279 }
280 myParsedRerouteInterval.main.push_back(edge);
281 myParsedRerouteInterval.cMain.push_back(edge);
282 }
283 for (const std::string& edgeID : attrs.get<std::vector<std::string> >(SUMO_ATTR_SIDING, getID().c_str(), ok)) {
284 MSEdge* edge = MSEdge::dictionary(edgeID);
285 if (edge == nullptr) {
286 throw InvalidArgument("The siding edge '" + edgeID + "' to use within rerouter '" + getID() + "' is not known.");
287 }
288 myParsedRerouteInterval.siding.push_back(edge);
289 myParsedRerouteInterval.cSiding.push_back(edge);
290 }
292 if (myParsedRerouteInterval.sidingExit == nullptr) {
293 throw InvalidArgument("The siding within rerouter '" + getID() + "' does not have a rail signal.");
294 }
295 }
296}
297
298
299void
301 if (element == SUMO_TAG_INTERVAL) {
302 for (auto paVi : myParsedRerouteInterval.parkProbs.getVals()) {
303 dynamic_cast<MSParkingArea*>(paVi.first)->setNumAlternatives((int)myParsedRerouteInterval.parkProbs.getVals().size() - 1);
304 }
305 if (myParsedRerouteInterval.closedLanes.size() > 0) {
306 // collect edges that are affect by a closed lane
307 std::set<MSEdge*> affected;
308 for (const MSLane* const l : myParsedRerouteInterval.closedLanes) {
309 affected.insert(&l->getEdge());
310 }
312 }
313 const SUMOTime closingBegin = myParsedRerouteInterval.begin;
314 const SUMOTime simBegin = string2time(OptionsCont::getOptions().getString("begin"));
315 if (closingBegin < simBegin && myParsedRerouteInterval.end > simBegin) {
316 // interval started before simulation begin but is still active at
317 // the start of the simulation
319 }
321 myIntervals.back().id = (long long int)&myIntervals.back();
325 }
326 }
327}
328
329
330// ------------ loading end
331
332
335 bool updateVehicles = false;
336 for (const RerouteInterval& i : myIntervals) {
337 if (i.begin == currentTime && !(i.closed.empty() && i.closedLanes.empty()) && i.permissions != SVCAll) {
338 for (MSEdge* const e : i.closed) {
339 for (MSLane* lane : e->getLanes()) {
340 //std::cout << SIMTIME << " closing: intervalID=" << i.id << " lane=" << lane->getID() << " prevPerm=" << getVehicleClassNames(lane->getPermissions()) << " new=" << getVehicleClassNames(i.permissions) << "\n";
341 lane->setPermissions(i.permissions, i.id);
342 }
343 e->rebuildAllowedLanes();
344 updateVehicles = true;
345 }
346 for (MSLane* const lane : i.closedLanes) {
347 lane->setPermissions(i.permissions, i.id);
348 lane->getEdge().rebuildAllowedLanes();
349 updateVehicles = true;
350 }
353 }
354 if (i.end == currentTime && !(i.closed.empty() && i.closedLanes.empty()) && i.permissions != SVCAll) {
355 for (MSEdge* const e : i.closed) {
356 for (MSLane* lane : e->getLanes()) {
357 lane->resetPermissions(i.id);
358 //std::cout << SIMTIME << " opening: intervalID=" << i.id << " lane=" << lane->getID() << " restore prevPerm=" << getVehicleClassNames(lane->getPermissions()) << "\n";
359 }
360 e->rebuildAllowedLanes();
361 updateVehicles = true;
362 }
363 for (MSLane* lane : i.closedLanes) {
364 lane->resetPermissions(i.id);
365 lane->getEdge().rebuildAllowedLanes();
366 updateVehicles = true;
367 }
368 }
369 }
370 if (updateVehicles) {
371 // only vehicles on the affected lanes had their bestlanes updated so far
372 for (MSEdge* e : myEdges) {
373 // also updates vehicles
374 e->rebuildAllowedTargets();
375 }
376 }
377 return 0;
378}
379
380
383 for (const RerouteInterval& ri : myIntervals) {
384 if (ri.begin <= time && ri.end > time) {
385 if (
386 // destProbReroute
387 ri.edgeProbs.getOverallProb() > 0 ||
388 // routeProbReroute
389 ri.routeProbs.getOverallProb() > 0 ||
390 // parkingZoneReroute
391 ri.parkProbs.getOverallProb() > 0) {
392 return &ri;
393 }
394 if (!ri.closed.empty() || !ri.closedLanesAffected.empty() || !ri.main.empty()) {
395 const std::set<SUMOTrafficObject::NumericalID>& edgeIndices = obj.getUpcomingEdgeIDs();
396 if (affected(edgeIndices, ri.closed)
397 || affected(edgeIndices, ri.closedLanesAffected)
398 || affected(edgeIndices, ri.main)) {
399 return &ri;
400 }
401 }
402 }
403 }
404 return nullptr;
405}
406
407
410 for (const RerouteInterval& ri : myIntervals) {
411 if (ri.begin <= time && ri.end > time) {
412 if (ri.edgeProbs.getOverallProb() != 0 || ri.routeProbs.getOverallProb() != 0 || ri.parkProbs.getOverallProb() != 0
413 || !ri.closed.empty() || !ri.closedLanesAffected.empty() || !ri.main.empty()) {
414 return &ri;
415 }
416 }
417 }
418 return nullptr;
419}
420
421
422bool
424 if (myAmOptional || myRadius != std::numeric_limits<double>::max()) {
425 return true;
426 }
427 return triggerRouting(tObject, reason);
428}
429
430
431bool
433 double /*newPos*/, double /*newSpeed*/) {
435}
436
437
438bool
440 MSMoveReminder::Notification reason, const MSLane* /* enteredLane */) {
441 return reason == NOTIFICATION_LANE_CHANGE;
442}
443
444
445bool
447 if (!applies(tObject)) {
448 return false;
449 }
450 if (myRadius != std::numeric_limits<double>::max() && tObject.getPosition().distanceTo(myPosition) > myRadius) {
451 return true;
452 }
453 // check whether the vehicle shall be rerouted
455 const MSTriggeredRerouter::RerouteInterval* const rerouteDef = getCurrentReroute(now, tObject);
456 if (rerouteDef == nullptr) {
457 return true; // an active interval could appear later
458 }
459 const double prob = myAmInUserMode ? myUserProbability : myProbability;
460 if (prob < 1 && RandHelper::rand(tObject.getRNG()) > prob) {
461 return false; // XXX another interval could appear later but we would have to track whether the current interval was already tried
462 }
463 if (myTimeThreshold > 0 && MAX2(tObject.getWaitingTime(), tObject.getWaitingTime(true)) < myTimeThreshold) {
464 return true; // waiting time may be reached later
465 }
466 if (reason == NOTIFICATION_LANE_CHANGE) {
467 return false;
468 }
469 // if we have a closingLaneReroute, only vehicles with a rerouting device can profit from rerouting (otherwise, edge weights will not reflect local jamming)
470 const bool hasReroutingDevice = tObject.getDevice(typeid(MSDevice_Routing)) != nullptr;
471 if (rerouteDef->closedLanes.size() > 0 && !hasReroutingDevice) {
472 return true; // an active interval could appear later
473 }
474 const MSEdge* lastEdge = tObject.getRerouteDestination();
475#ifdef DEBUG_REROUTER
476 if (DEBUGCOND) {
477 std::cout << SIMTIME << " veh=" << veh.getID() << " check rerouter " << getID() << " lane=" << Named::getIDSecure(veh.getLane()) << " edge=" << veh.getEdge()->getID() << " finalEdge=" << lastEdge->getID() << " arrivalPos=" << veh.getArrivalPos() << "\n";
478 }
479#endif
480
481 if (rerouteDef->parkProbs.getOverallProb() > 0) {
482#ifdef HAVE_FOX
483 ScopedLocker<> lock(myNotificationMutex, MSGlobals::gNumSimThreads > 1);
484#endif
485 if (!tObject.isVehicle()) {
486 return false;
487 }
488 SUMOVehicle& veh = static_cast<SUMOVehicle&>(tObject);
489 bool newDestination = false;
490 ConstMSEdgeVector newRoute;
491 MSParkingArea* newParkingArea = rerouteParkingArea(rerouteDef, veh, newDestination, newRoute);
492 if (newParkingArea != nullptr) {
493 // adapt plans of any riders
494 for (MSTransportable* p : veh.getPersons()) {
495 p->rerouteParkingArea(veh.getNextParkingArea(), newParkingArea);
496 }
497
498 if (newDestination) {
499 // update arrival parameters
500 SUMOVehicleParameter* newParameter = new SUMOVehicleParameter();
501 *newParameter = veh.getParameter();
503 newParameter->arrivalPos = newParkingArea->getEndLanePosition();
504 veh.replaceParameter(newParameter);
505 }
506
507 SUMOAbstractRouter<MSEdge, SUMOVehicle>& router = hasReroutingDevice
508 ? MSRoutingEngine::getRouterTT(veh.getRNGIndex(), veh.getVClass(), rerouteDef->closed)
509 : MSNet::getInstance()->getRouterTT(veh.getRNGIndex(), rerouteDef->closed);
510 const double routeCost = router.recomputeCosts(newRoute, &veh, MSNet::getInstance()->getCurrentTimeStep());
511 ConstMSEdgeVector prevEdges(veh.getCurrentRouteEdge(), veh.getRoute().end());
512 const double previousCost = router.recomputeCosts(prevEdges, &veh, MSNet::getInstance()->getCurrentTimeStep());
513 const double savings = previousCost - routeCost;
514 hasReroutingDevice
516 : MSNet::getInstance()->getRouterTT(veh.getRNGIndex()); // reset closed edges
517 //if (getID() == "ego") std::cout << SIMTIME << " pCost=" << previousCost << " cost=" << routeCost
518 // << " prevEdges=" << toString(prevEdges)
519 // << " newEdges=" << toString(edges)
520 // << "\n";
521
522 std::string errorMsg;
523 if (veh.replaceParkingArea(newParkingArea, errorMsg)) {
524 veh.replaceRouteEdges(newRoute, routeCost, savings, getID() + ":" + toString(SUMO_TAG_PARKING_AREA_REROUTE), false, false, false);
525 } else {
526 WRITE_WARNING("Vehicle '" + veh.getID() + "' at rerouter '" + getID()
527 + "' could not reroute to new parkingArea '" + newParkingArea->getID()
528 + "' reason=" + errorMsg + ", time=" + time2string(MSNet::getInstance()->getCurrentTimeStep()) + ".");
529 }
530 }
531 return false;
532 }
533 if (rerouteDef->main.size() > 0) {
534 if (!tObject.isVehicle()) {
535 return false;
536 }
537 SUMOVehicle& veh = static_cast<SUMOVehicle&>(tObject);
538 const ConstMSEdgeVector& oldEdges = veh.getRoute().getEdges();
539 auto mainStart = std::find(veh.getCurrentRouteEdge(), oldEdges.end(), rerouteDef->main.front());
540 if (mainStart == oldEdges.end()
541 // exit main within
542 || ConstMSEdgeVector(mainStart, mainStart + rerouteDef->main.size()) != rerouteDef->cMain
543 // stop in main
544 || (veh.hasStops() && veh.getNextStop().edge < (mainStart + rerouteDef->main.size()))) {
545 //std::cout << SIMTIME << " veh=" << veh.getID() << " wrong route or stop\n";
546 return false;
547 }
548 std::pair<const SUMOVehicle*, MSRailSignal*> overtaker_signal = overtakingTrain(veh, mainStart, rerouteDef->main);
549 if (overtaker_signal.first != nullptr) {
550 SUMOAbstractRouter<MSEdge, SUMOVehicle>& router = hasReroutingDevice
551 ? MSRoutingEngine::getRouterTT(veh.getRNGIndex(), veh.getVClass(), rerouteDef->closed)
552 : MSNet::getInstance()->getRouterTT(veh.getRNGIndex(), rerouteDef->closed);
553 ConstMSEdgeVector newEdges(veh.getCurrentRouteEdge(), mainStart);
554 newEdges.insert(newEdges.end(), rerouteDef->siding.begin(), rerouteDef->siding.end());
555 newEdges.insert(newEdges.end(), mainStart + rerouteDef->main.size(), oldEdges.end());
556 const double routeCost = router.recomputeCosts(newEdges, &veh, MSNet::getInstance()->getCurrentTimeStep());
557 const double savings = (router.recomputeCosts(rerouteDef->cMain, &veh, MSNet::getInstance()->getCurrentTimeStep())
558 - router.recomputeCosts(rerouteDef->cSiding, &veh, MSNet::getInstance()->getCurrentTimeStep()));
559 const std::string info = getID() + ":" + toString(SUMO_TAG_OVERTAKING_REROUTE) + ":" + overtaker_signal.first->getID();
560 veh.replaceRouteEdges(newEdges, routeCost, savings, info, false, false, false);
562 MSRailSignalConstraint::PREDECESSOR, overtaker_signal.second, overtaker_signal.first->getID(), 100, true));
563 }
564 return false;
565 }
566
567 // get rerouting params
568 ConstMSRoutePtr newRoute = rerouteDef->routeProbs.getOverallProb() > 0 ? rerouteDef->routeProbs.get() : nullptr;
569 // we will use the route if given rather than calling our own dijsktra...
570 if (newRoute != nullptr) {
571#ifdef DEBUG_REROUTER
572 if (DEBUGCOND) {
573 std::cout << " replacedRoute from routeDist " << newRoute->getID() << "\n";
574 }
575#endif
576 tObject.replaceRoute(newRoute, getID());
577 return false; // XXX another interval could appear later but we would have to track whether the currenty interval was already used
578 }
579 const MSEdge* newEdge = lastEdge;
580 // ok, try using a new destination
581 double newArrivalPos = -1;
582 const bool destUnreachable = std::find(rerouteDef->closed.begin(), rerouteDef->closed.end(), lastEdge) != rerouteDef->closed.end();
583 bool keepDestination = false;
584 // if we have a closingReroute, only assign new destinations to vehicles which cannot reach their original destination
585 // if we have a closingLaneReroute, no new destinations should be assigned
586 if (rerouteDef->closed.empty() || destUnreachable || rerouteDef->isVia) {
587 newEdge = rerouteDef->edgeProbs.getOverallProb() > 0 ? rerouteDef->edgeProbs.get() : lastEdge;
588 assert(newEdge != nullptr);
589 if (newEdge == &mySpecialDest_terminateRoute) {
590 keepDestination = true;
591 newEdge = tObject.getEdge();
592 newArrivalPos = tObject.getPositionOnLane(); // instant arrival
593 } else if (newEdge == &mySpecialDest_keepDestination || newEdge == lastEdge) {
594 if (destUnreachable && rerouteDef->permissions == SVCAll) {
595 // if permissions aren't set vehicles will simply drive through
596 // the closing unless terminated. If the permissions are specified, assume that the user wants
597 // vehicles to stand and wait until the closing ends
598 WRITE_WARNINGF(TL("Cannot keep destination edge '%' for vehicle '%' due to closed edges. Terminating route."), lastEdge->getID(), tObject.getID());
599 newEdge = tObject.getEdge();
600 } else {
601 newEdge = lastEdge;
602 }
603 }
604 }
605 ConstMSEdgeVector edges;
606 std::vector<MSTransportableRouter::TripItem> items;
607 // we have a new destination, let's replace the route (if it is affected)
608 if (rerouteDef->closed.empty() || destUnreachable || rerouteDef->isVia || affected(tObject.getUpcomingEdgeIDs(), rerouteDef->closed)) {
609 if (tObject.isVehicle()) {
610 SUMOVehicle& veh = static_cast<SUMOVehicle&>(tObject);
611 const bool canChangeDest = rerouteDef->edgeProbs.getOverallProb() > 0;
612 MSVehicleRouter& router = hasReroutingDevice
613 ? MSRoutingEngine::getRouterTT(veh.getRNGIndex(), veh.getVClass(), rerouteDef->closed)
614 : MSNet::getInstance()->getRouterTT(veh.getRNGIndex(), rerouteDef->closed);
615 bool ok = veh.reroute(now, getID(), router, false, false, canChangeDest, newEdge);
616 if (!ok && !keepDestination && canChangeDest) {
617 // destination unreachable due to closed intermediate edges. pick among alternative targets
618 RandomDistributor<MSEdge*> edgeProbs2 = rerouteDef->edgeProbs;
619 edgeProbs2.remove(const_cast<MSEdge*>(newEdge));
620 while (!ok && edgeProbs2.getVals().size() > 0) {
621 newEdge = edgeProbs2.get();
622 edgeProbs2.remove(const_cast<MSEdge*>(newEdge));
623 if (newEdge == &mySpecialDest_terminateRoute) {
624 newEdge = veh.getEdge();
625 newArrivalPos = veh.getPositionOnLane(); // instant arrival
626 }
627 if (newEdge == &mySpecialDest_keepDestination && rerouteDef->permissions != SVCAll) {
628 newEdge = lastEdge;
629 break;
630 }
631 ok = veh.reroute(now, getID(), router, false, false, true, newEdge);
632 }
633
634 }
635 if (!rerouteDef->isVia) {
636#ifdef DEBUG_REROUTER
637 if (DEBUGCOND) std::cout << " rerouting: newDest=" << newEdge->getID()
638 << " newEdges=" << toString(edges)
639 << " useNewRoute=" << useNewRoute << " newArrivalPos=" << newArrivalPos << " numClosed=" << rerouteDef->closed.size()
640 << " destUnreachable=" << destUnreachable << " containsClosed=" << veh.getRoute().containsAnyOf(rerouteDef->closed) << "\n";
641#endif
642 if (ok && newArrivalPos != -1) {
643 // must be called here because replaceRouteEdges may also set the arrivalPos
644 veh.setArrivalPos(newArrivalPos);
645 }
646
647 }
648 } else {
649 // person rerouting here
650 MSTransportableRouter& router = hasReroutingDevice
652 : MSNet::getInstance()->getIntermodalRouter(tObject.getRNGIndex(), 0, rerouteDef->closed);
653 const bool success = router.compute(tObject.getEdge(), newEdge, tObject.getPositionOnLane(), "",
654 rerouteDef->isVia ? newEdge->getLength() / 2. : tObject.getParameter().arrivalPos, "",
655 tObject.getMaxSpeed(), nullptr, 0, now, items);
656 if (!rerouteDef->isVia) {
657 if (success) {
658 for (const MSTransportableRouter::TripItem& it : items) {
659 if (!it.edges.empty() && !edges.empty() && edges.back() == it.edges.front()) {
660 edges.pop_back();
661 }
662 edges.insert(edges.end(), std::make_move_iterator(it.edges.begin()), std::make_move_iterator(it.edges.end()));
663 if (!edges.empty()) {
664 static_cast<MSPerson&>(tObject).replaceWalk(edges, tObject.getPositionOnLane(), 0, 1);
665 }
666 }
667 } else {
668 // maybe the pedestrian model still finds a way (JuPedSim)
669 static_cast<MSPerson&>(tObject).replaceWalk({tObject.getEdge(), newEdge}, tObject.getPositionOnLane(), 0, 1);
670 }
671 }
672 }
673 }
674 // it was only a via so calculate the remaining part
675 if (rerouteDef->isVia) {
676 if (tObject.isVehicle()) {
677 SUMOVehicle& veh = static_cast<SUMOVehicle&>(tObject);
678 if (!edges.empty()) {
679 edges.pop_back();
680 }
681 MSVehicleRouter& router = hasReroutingDevice
682 ? MSRoutingEngine::getRouterTT(veh.getRNGIndex(), veh.getVClass(), rerouteDef->closed)
683 : MSNet::getInstance()->getRouterTT(veh.getRNGIndex(), rerouteDef->closed);
684 router.compute(newEdge, lastEdge, &veh, now, edges);
685 const double routeCost = router.recomputeCosts(edges, &veh, now);
686 hasReroutingDevice
688 : MSNet::getInstance()->getRouterTT(veh.getRNGIndex()); // reset closed edges
689 const bool useNewRoute = veh.replaceRouteEdges(edges, routeCost, 0, getID());
690#ifdef DEBUG_REROUTER
691 if (DEBUGCOND) std::cout << " rerouting: newDest=" << newEdge->getID()
692 << " newEdges=" << toString(edges)
693 << " useNewRoute=" << useNewRoute << " newArrivalPos=" << newArrivalPos << " numClosed=" << rerouteDef->closed.size()
694 << " destUnreachable=" << destUnreachable << " containsClosed=" << veh.getRoute().containsAnyOf(rerouteDef->closed) << "\n";
695#endif
696 if (useNewRoute && newArrivalPos != -1) {
697 // must be called here because replaceRouteEdges may also set the arrivalPos
698 veh.setArrivalPos(newArrivalPos);
699 }
700 } else {
701 // person rerouting here
702 bool success = !items.empty();
703 if (success) {
704 MSTransportableRouter& router = hasReroutingDevice
706 : MSNet::getInstance()->getIntermodalRouter(tObject.getRNGIndex(), 0, rerouteDef->closed);
707 success = router.compute(newEdge, lastEdge, newEdge->getLength() / 2., "",
708 tObject.getParameter().arrivalPos, "",
709 tObject.getMaxSpeed(), nullptr, 0, now, items);
710 }
711 if (success) {
712 for (const MSTransportableRouter::TripItem& it : items) {
713 if (!it.edges.empty() && !edges.empty() && edges.back() == it.edges.front()) {
714 edges.pop_back();
715 }
716 edges.insert(edges.end(), std::make_move_iterator(it.edges.begin()), std::make_move_iterator(it.edges.end()));
717 }
718 if (!edges.empty()) {
719 static_cast<MSPerson&>(tObject).replaceWalk(edges, tObject.getPositionOnLane(), 0, 1);
720 }
721 } else {
722 // maybe the pedestrian model still finds a way (JuPedSim)
723 static_cast<MSPerson&>(tObject).replaceWalk({tObject.getEdge(), newEdge, lastEdge}, tObject.getPositionOnLane(), 0, 1);
724 }
725 }
726 }
727 return false; // XXX another interval could appear later but we would have to track whether the currenty interval was already used
728}
729
730
731void
735
736
737void
741
742
743bool
747
748
749double
753
754
755double
759
760
761double
763 return dynamic_cast<MSParkingArea*>(parkingArea)->getOccupancy();
764}
765
766
767double
769 return dynamic_cast<MSParkingArea*>(parkingArea)->getLastStepOccupancy();
770}
771
772
773double
775 MSParkingArea* pa = dynamic_cast<MSParkingArea*>(parkingArea);
776 return pa->getCapacity();
777}
778
779
780void
782 veh.rememberBlockedParkingArea(parkingArea, blocked);
783}
784
785
786void
788 veh.rememberParkingAreaScore(parkingArea, score);
789}
790
791
792void
796
797
800 return veh.sawBlockedParkingArea(parkingArea, local);
801}
802
803
804int
808
809
810void
814
815
818 SUMOVehicle& veh, bool& newDestination, ConstMSEdgeVector& newRoute) {
819 std::vector<StoppingPlaceVisible> parks = rerouteDef->parkProbs.getVals();
820 StoppingPlaceParamMap_t addInput = {};
821 return dynamic_cast<MSParkingArea*>(rerouteStoppingPlace(parks, rerouteDef->parkProbs.getProbs(), veh, newDestination, newRoute, addInput, rerouteDef->closed));
822}
823
824
825std::pair<const SUMOVehicle*, MSRailSignal*>
826MSTriggeredRerouter::overtakingTrain(const SUMOVehicle& veh, ConstMSEdgeVector::const_iterator mainStart, const MSEdgeVector& main) {
827 const double vMax = veh.getMaxSpeed();
829 for (MSVehicleControl::constVehIt it_veh = c.loadedVehBegin(); it_veh != c.loadedVehEnd(); ++it_veh) {
830 const SUMOVehicle* veh2 = (*it_veh).second;
831 if (veh2->isOnRoad() && veh2->getMaxSpeed() > vMax) {
832 const ConstMSEdgeVector& route2 = veh2->getRoute().getEdges();
833 auto itOnMain2 = route2.end();
834 int mainIndex = 0;
835 for (const MSEdge* m : main) {
836 itOnMain2 = std::find(route2.begin(), route2.end(), m);
837 if (itOnMain2 != route2.end()) {
838 break;
839 }
840 mainIndex++;
841 }
842 if (itOnMain2 != route2.end()) {
843 auto itOnMain = mainStart + mainIndex;
844 double timeToMain = 0;
845 // veh2 may be anywhere on the current edge so we have to discount
846 double timeToMain2 = -veh2->getEdge()->getMinimumTravelTime(veh2) * veh2->getPositionOnLane() / veh2->getEdge()->getLength();
847 for (auto it = veh.getCurrentRouteEdge(); it != itOnMain; it++) {
848 timeToMain += (*it)->getMinimumTravelTime(&veh);
849 }
850 for (auto it = veh2->getCurrentRouteEdge(); it != itOnMain2; it++) {
851 timeToMain2 += (*it)->getMinimumTravelTime(veh2);
852 }
853 double commonTime = 0;
854 double commonTime2 = 0;
855 int nCommon = 0;
856 auto exitMain2 = itOnMain2;
857 while (itOnMain2 != route2.end() && *itOnMain == *itOnMain2) {
858 const MSEdge* common = *itOnMain;
859 commonTime += common->getMinimumTravelTime(&veh);
860 commonTime2 += common->getMinimumTravelTime(veh2);
861 nCommon++;
862 itOnMain++;
863 itOnMain2++;
864 }
865 exitMain2 += MIN2(nCommon, (int)main.size() - mainIndex);
866 const double saving = timeToMain + commonTime - (timeToMain2 + commonTime2);
867 //std::cout << " veh=" << veh.getID() << " veh2=" << veh2->getID() << " nCommon=" << nCommon << " cT=" << commonTime << " cT2=" << commonTime2
868 // << " ttm=" << timeToMain << " ttm2=" << timeToMain2 << " saving=" << saving << "\n";
869 if (saving > 300) {
870 MSRailSignal* s = findSignal(veh2->getCurrentRouteEdge(), exitMain2);
871 if (s != nullptr) {
872 return std::make_pair(veh2, s);
873 }
874 }
875 }
876 }
877 }
878 return std::make_pair(nullptr, nullptr);
879}
880
881
883MSTriggeredRerouter::findSignal(ConstMSEdgeVector::const_iterator begin, ConstMSEdgeVector::const_iterator end) {
884 auto it = end;
885 do {
886 it--;
887 const MSEdge* edge = *it;
889 for (const MSLink* link : edge->getLanes().front()->getLinkCont()) {
890 if (link->getTLLogic() != nullptr) {
891 return dynamic_cast<MSRailSignal*>(const_cast<MSTrafficLightLogic*>(link->getTLLogic()));
892 }
893 }
894 }
895 } while (it != begin);
896 return nullptr;
897}
898
899bool
901 if (myVehicleTypes.empty() || myVehicleTypes.count(obj.getVehicleType().getOriginalID()) > 0) {
902 return true;
903 } else {
905 for (auto vTypeDist : vTypeDists) {
906 if (myVehicleTypes.count(vTypeDist) > 0) {
907 return true;
908 }
909 }
910 return false;
911 }
912}
913
914
915bool
916MSTriggeredRerouter::affected(const std::set<SUMOTrafficObject::NumericalID>& edgeIndices, const MSEdgeVector& closed) {
917 for (const MSEdge* const e : closed) {
918 if (edgeIndices.count(e->getNumericalID()) > 0) {
919 return true;
920 }
921 }
922 return false;
923}
924
925
926void
928 // if a parkingArea is a rerouting target, it should generally have a
929 // rerouter on its edge or vehicles will be stuck there once it's full.
930 // The user should receive a Warning in this case
931 std::set<MSEdge*> parkingRerouterEdges;
932 std::map<MSParkingArea*, std::string, ComparatorIdLess> targetedParkingArea; // paID -> targetingRerouter
933 for (const auto& rr : myInstances) {
934 bool hasParkingReroute = false;
935 for (const RerouteInterval& interval : rr.second->myIntervals) {
936 if (interval.parkProbs.getOverallProb() > 0) {
937 hasParkingReroute = true;
938 for (const StoppingPlaceVisible& pav : interval.parkProbs.getVals()) {
939 targetedParkingArea[dynamic_cast<MSParkingArea*>(pav.first)] = rr.first;
940 }
941 }
942 }
943 if (hasParkingReroute) {
944 parkingRerouterEdges.insert(rr.second->myEdges.begin(), rr.second->myEdges.end());
945 }
946 }
947 for (const auto& item : targetedParkingArea) {
948 if (parkingRerouterEdges.count(&item.first->getLane().getEdge()) == 0) {
949 WRITE_WARNINGF(TL("ParkingArea '%' is targeted by rerouter '%' but doesn't have its own rerouter. This may cause parking search to abort."),
950 item.first->getID(), item.second);
951 }
952 }
953}
954
955
956/****************************************************************************/
long long int SUMOTime
Definition GUI.h:36
std::vector< const MSEdge * > ConstMSEdgeVector
Definition MSEdge.h:74
std::vector< MSEdge * > MSEdgeVector
Definition MSEdge.h:73
#define DEBUGCOND(PED)
#define WRITE_WARNINGF(...)
Definition MsgHandler.h:284
#define WRITE_WARNING(msg)
Definition MsgHandler.h:283
#define TL(string)
Definition MsgHandler.h:301
#define TLF(string,...)
Definition MsgHandler.h:303
std::shared_ptr< const MSRoute > ConstMSRoutePtr
Definition Route.h:32
SUMOTime string2time(const std::string &r)
convert string to SUMOTime
Definition SUMOTime.cpp:46
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 SIMTIME
Definition SUMOTime.h:62
const SVCPermissions SVCAll
all VClasses are allowed
SVCPermissions parseVehicleClasses(const std::string &allowedS)
Parses the given definition of allowed vehicle classes into the given containers Deprecated classes g...
@ SVC_AUTHORITY
authorities vehicles
@ GIVEN
The arrival position is given.
@ SUMO_TAG_INTERVAL
an aggreagated-output interval
@ SUMO_TAG_CLOSING_REROUTE
reroute of type closing
@ SUMO_TAG_PARKING_AREA_REROUTE
entry for an alternative parking zone
@ SUMO_TAG_PARKING_AREA
A parking area.
@ SUMO_TAG_ROUTE_PROB_REROUTE
probability of route of a reroute
@ SUMO_TAG_VIA_PROB_REROUTE
probability of a via reroute
@ SUMO_TAG_DEST_PROB_REROUTE
probability of destination of a reroute
@ SUMO_TAG_OVERTAKING_REROUTE
decision point for rerouting to be overtaken
@ SUMO_TAG_CLOSING_LANE_REROUTE
lane of a reroute of type closing
@ SUMO_ATTR_DISALLOW
@ SUMO_ATTR_ALLOW
@ SUMO_ATTR_MAIN
@ SUMO_ATTR_BEGIN
weights: time range begin
@ SUMO_ATTR_END
weights: time range end
@ SUMO_ATTR_PROB
@ SUMO_ATTR_SIDING
@ SUMO_ATTR_ID
@ SUMO_ATTR_VISIBLE
T MIN2(T a, T b)
Definition StdDefs.h:76
T MAX2(T a, T b)
Definition StdDefs.h:82
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:46
int main(int argc, char *argv[])
bool compute(const E *from, const E *to, const double departPos, const std::string &originStopID, const double arrivalPos, const std::string &stopID, const double speed, const V *const vehicle, const SVCPermissions modeSet, const SUMOTime msTime, std::vector< TripItem > &into, const double externalFactor=0.)
Builds the route between the given edges using the minimum effort at the given time The definition of...
MESegment * getSegmentForEdge(const MSEdge &e, double pos=0)
Get the segment for a given edge at a given position.
Definition MELoop.cpp:340
void addDetector(MSMoveReminder *data, int queueIndex=-1)
Adds a data collector for a detector to this segment.
A device that performs vehicle rerouting based on current edge speeds.
A road/street connecting two junctions.
Definition MSEdge.h:77
const std::vector< MSLane * > & getLanes() const
Returns this edge's lanes.
Definition MSEdge.h:168
const MSJunction * getToJunction() const
Definition MSEdge.h:418
double getLength() const
return the length of the edge
Definition MSEdge.h:685
double getMinimumTravelTime(const SUMOVehicle *const veh) const
returns the minimum travel time for the given vehicle
Definition MSEdge.h:476
static bool dictionary(const std::string &id, MSEdge *edge)
Inserts edge into the static dictionary Returns true if the key id isn't already in the dictionary....
Definition MSEdge.cpp:1047
virtual void addEvent(Command *operation, SUMOTime execTimeStep=-1)
Adds an Event.
static bool gUseMesoSim
Definition MSGlobals.h:106
static MELoop * gMesoNet
mesoscopic simulation infrastructure
Definition MSGlobals.h:112
static int gNumSimThreads
how many threads to use for simulation
Definition MSGlobals.h:146
SumoXMLNodeType getType() const
return the type of this Junction
Definition MSJunction.h:133
Representation of a lane in the micro simulation.
Definition MSLane.h:84
static bool dictionary(const std::string &id, MSLane *lane)
Static (sic!) container methods {.
Definition MSLane.cpp:2451
Something on a lane to be noticed about vehicle movement.
Notification
Definition of a vehicle state.
@ NOTIFICATION_LANE_CHANGE
The vehicle changes lanes (micro 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:186
MSEventControl * getBeginOfTimestepEvents()
Returns the event control for events executed at the begin of a time step.
Definition MSNet.h:472
MSTransportableRouter & getIntermodalRouter(const int rngIndex, const int routingMode=0, const MSEdgeVector &prohibited=MSEdgeVector()) const
Definition MSNet.cpp:1561
SUMOTime getCurrentTimeStep() const
Returns the current simulation step.
Definition MSNet.h:321
MSStoppingPlace * getStoppingPlace(const std::string &id, const SumoXMLTag category) const
Returns the named stopping place of the given category.
Definition MSNet.cpp:1396
MSVehicleRouter & getRouterTT(const int rngIndex, const MSEdgeVector &prohibited=MSEdgeVector()) const
Definition MSNet.cpp:1523
MSVehicleControl & getVehicleControl()
Returns the vehicle control.
Definition MSNet.h:379
A lane area vehicles can halt at.
int getCapacity() const
Returns the area capacity.
A signal for rails.
void addConstraint(const std::string &tripId, MSRailSignalConstraint *constraint)
register constraint for signal switching
const ConstMSEdgeVector & getEdges() const
Definition MSRoute.h:125
MSRouteIterator end() const
Returns the end of the list of edges to pass.
Definition MSRoute.cpp:79
bool containsAnyOf(const MSEdgeVector &edgelist) const
Definition MSRoute.cpp:243
static bool dictionary(const std::string &id, ConstMSRoutePtr route)
Adds a route to the dictionary.
Definition MSRoute.cpp:109
static MSVehicleRouter & getRouterTT(const int rngIndex, SUMOVehicleClass svc, const MSEdgeVector &prohibited=MSEdgeVector())
return the vehicle router instance
static MSTransportableRouter & getIntermodalRouterTT(const int rngIndex, const MSEdgeVector &prohibited=MSEdgeVector())
return the person router instance
MSRouteIterator edge
The edge in the route to stop at.
Definition MSStop.h:48
A lane area vehicles can halt at.
double getEndLanePosition() const
Returns the end position of this stop.
std::map< std::string, double > StoppingPlaceParamMap_t
MSStoppingPlace * rerouteStoppingPlace(std::vector< StoppingPlaceVisible > &stoppingPlaceCandidates, const std::vector< double > &probs, SUMOVehicle &veh, bool &newDestination, ConstMSEdgeVector &newRoute, StoppingPlaceParamMap_t &scores, const MSEdgeVector &closedEdges={}, const int insertStopIndex=0, const bool keepCurrentStop=true)
main method to trigger the rerouting to the "best" StoppingPlace according to the custom evaluation f...
std::pair< MSStoppingPlace *, bool > StoppingPlaceVisible
The parent class for traffic light logics.
SUMOTime setPermissions(const SUMOTime currentTime)
Sets the edge permission if there are any defined in the closingEdge.
double getStoppingPlaceCapacity(MSStoppingPlace *parkingArea)
Return the number of places the ParkingArea provides.
bool notifyEnter(SUMOTrafficObject &veh, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Tries to reroute the vehicle.
int getNumberStoppingPlaceReroutes(SUMOVehicle &veh)
ask how many times already the vehicle has been rerouted to another stopping place
bool notifyMove(SUMOTrafficObject &veh, double oldPos, double newPos, double newSpeed)
Triggers rerouting (once) for vehicles that are already on the edge when the rerouter activates.
MSRailSignal * findSignal(ConstMSEdgeVector::const_iterator begin, ConstMSEdgeVector::const_iterator end)
find the last downstream signal on the given route
bool notifyLeave(SUMOTrafficObject &veh, double lastPos, MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
Removes the reminder.
MSTriggeredRerouter(const std::string &id, const MSEdgeVector &edges, double prob, bool off, bool optional, SUMOTime timeThreshold, const std::string &vTypes, const Position &pos, const double radius)
Constructor.
double getUserProbability() const
Returns the rerouting probability given by the user.
MSParkingArea * rerouteParkingArea(const MSTriggeredRerouter::RerouteInterval *rerouteDef, SUMOVehicle &veh, bool &newDestination, ConstMSEdgeVector &newRoute)
search for an alternative ParkingArea
bool myHaveParkProbs
whether this rerouter has loaded parkingReroute definitions
virtual void myStartElement(int element, const SUMOSAXAttributes &attrs)
Called on the opening of a tag;.
bool myAmOptional
Information whether the rerouting will only take place on request.
static bool affected(const std::set< SUMOTrafficObject::NumericalID > &edgeIndices, const MSEdgeVector &closed)
void setUserUsageProbability(double prob)
Sets the probability with which a vehicle is rerouted given by the user.
Position myPosition
Where are we located in the network.
void rememberStoppingPlaceScore(SUMOVehicle &veh, MSStoppingPlace *parkingArea, const std::string &score)
store the score of the ParkingArea in the vehicle
void rememberBlockedStoppingPlace(SUMOVehicle &veh, const MSStoppingPlace *parkingArea, bool blocked)
store the blocked ParkingArea in the vehicle
static MSEdge mySpecialDest_terminateRoute
std::pair< const SUMOVehicle *, MSRailSignal * > overtakingTrain(const SUMOVehicle &veh, ConstMSEdgeVector::const_iterator mainStart, const MSEdgeVector &main)
determine whether veh should switch from main to siding to be overtaken and return the overtaking veh...
virtual void myEndElement(int element)
Called when a closing tag occurs.
bool triggerRouting(SUMOTrafficObject &veh, MSMoveReminder::Notification reason)
double myRadius
At which distance are we activated.
void resetStoppingPlaceScores(SUMOVehicle &veh)
reset all stored ParkingArea scores for this vehicle
double myProbability
The probability and the user-given probability.
SUMOTime sawBlockedStoppingPlace(SUMOVehicle &veh, MSStoppingPlace *parkingArea, bool local)
get the time the ParkingArea was considered full from this vehicle
virtual ~MSTriggeredRerouter()
Destructor.
bool inUserMode() const
Returns whether the user is setting the rerouting probability.
void setNumberStoppingPlaceReroutes(SUMOVehicle &veh, int value)
update the number of reroutes for the vehicle
std::set< std::string > myVehicleTypes
The vehicle types to look for (empty means all)
const RerouteInterval * getCurrentReroute(SUMOTime time, SUMOTrafficObject &obj) const
Returns the rerouting definition valid for the given time and object, nullptr if none.
static void checkParkingRerouteConsistency()
issues warning for incomplete parkingReroute relationships
static std::map< std::string, MSTriggeredRerouter * > myInstances
bool applies(const SUMOTrafficObject &obj) const
Checks whether the detector measures objects of the given type.
bool myAmInUserMode
Information whether the current rerouting probability is the user-given.
const MSEdgeVector myEdges
edges where vehicles are notified
double getLastStepStoppingPlaceOccupancy(MSStoppingPlace *parkingArea)
Return the number of occupied places of the StoppingPlace from the previous time step.
static MSEdge mySpecialDest_keepDestination
special destination values
RerouteInterval myParsedRerouteInterval
used during loading
double getStoppingPlaceOccupancy(MSStoppingPlace *parkingArea)
Return the number of occupied places of the ParkingArea.
double getProbability() const
Returns the rerouting probability.
std::vector< RerouteInterval > myIntervals
List of rerouting definition intervals.
void setUserMode(bool val)
Sets whether the process is currently steered by the user.
The class responsible for building and deletion of vehicles.
std::map< std::string, SUMOVehicle * >::const_iterator constVehIt
Definition of the internal vehicles map iterator.
constVehIt loadedVehBegin() const
Returns the begin of the internal vehicle map.
constVehIt loadedVehEnd() const
Returns the end of the internal vehicle map.
const std::set< std::string > getVTypeDistributionMembership(const std::string &id) const
Return the distribution IDs the vehicle type is a member of.
const std::string & getOriginalID() const
Returns the id of the original vehicle type if this is a vehicle specific type, the id otherwise.
Base class for objects which have an id.
Definition Named.h:54
static std::string getIDSecure(const T *obj, const std::string &fallBack="NULL")
get an identifier for Named-like object which may be Null
Definition Named.h:67
const std::string & getID() const
Returns the id.
Definition Named.h:74
static OptionsCont & getOptions()
Retrieves the options.
A point in 2D or 3D with translation and scaling methods.
Definition Position.h:37
static const Position INVALID
used to indicate that a position is valid
Definition Position.h:319
double distanceTo(const Position &p2) const
returns the euclidean distance in 3 dimensions
Definition Position.h:263
static double rand(SumoRNG *rng=nullptr)
Returns a random real number in [0, 1)
Represents a generic random distribution.
double getOverallProb() const
Return the sum of the probabilites assigned to the members.
T get(SumoRNG *which=nullptr) const
Draw a sample of the distribution.
bool add(T val, double prob, bool checkDuplicates=true)
Adds a value with an assigned probability to the distribution.
const std::vector< T > & getVals() const
Returns the members of the distribution.
bool remove(T val)
Removes a value with an assigned probability from the distribution.
const std::vector< double > & getProbs() const
Returns the probabilities assigned to the members of the distribution.
virtual bool compute(const E *from, const E *to, const V *const vehicle, SUMOTime msTime, std::vector< const E * > &into, bool silent=false)=0
Builds the route between the given edges using the minimum effort at the given time The definition of...
virtual double recomputeCosts(const std::vector< const E * > &edges, const V *const v, SUMOTime msTime, double *lengthp=nullptr) const
Encapsulated SAX-Attributes.
T getOpt(int attr, const char *objectid, bool &ok, T defaultValue=T(), bool report=true) const
Tries to read given attribute assuming it is an int.
SUMOTime getOptSUMOTimeReporting(int attr, const char *objectid, bool &ok, SUMOTime defaultValue, bool report=true) const
Tries to read given attribute assuming it is a SUMOTime.
virtual std::string getStringSecure(int id, const std::string &def) const =0
Returns the string-value of the named (by its enum-value) attribute.
T get(int attr, const char *objectid, bool &ok, bool report=true) const
Tries to read given attribute assuming it is an int.
virtual bool hasAttribute(int id) const =0
Returns the information whether the named (by its enum-value) attribute is within the current list.
Representation of a vehicle, person, or container.
virtual bool isVehicle() const
Whether it is a vehicle.
virtual const MSVehicleType & getVehicleType() const =0
Returns the object's "vehicle" type.
virtual int getRNGIndex() const =0
virtual MSDevice * getDevice(const std::type_info &type) const =0
Returns a device of the given type if it exists or nullptr if not.
virtual bool replaceRoute(ConstMSRoutePtr route, const std::string &info, bool onInit=false, int offset=0, bool addStops=true, bool removeStops=true, std::string *msgReturn=nullptr)=0
Replaces the current route by the given one.
virtual SUMOTime getWaitingTime(const bool accumulated=false) const =0
virtual const SUMOVehicleParameter & getParameter() const =0
Returns the vehicle's parameter (including departure definition)
virtual double getMaxSpeed() const =0
Returns the object's maximum speed (minimum of technical and desired maximum speed)
virtual SUMOVehicleClass getVClass() const =0
Returns the object's access class.
virtual SumoRNG * getRNG() const =0
Returns the associated RNG for this object.
virtual Position getPosition(const double offset=0) const =0
Return current position (x/y, cartesian)
virtual const std::set< NumericalID > getUpcomingEdgeIDs() const =0
returns the numerical IDs of edges to be used (possibly of future stages)
virtual const MSEdge * getEdge() const =0
Returns the edge the object is currently at.
virtual const MSEdge * getRerouteDestination() const =0
Returns the end point for reroutes (usually the last edge of the route)
virtual double getPositionOnLane() const =0
Get the object's position along the lane.
Representation of a vehicle.
Definition SUMOVehicle.h:62
virtual bool reroute(SUMOTime t, const std::string &info, SUMOAbstractRouter< MSEdge, SUMOVehicle > &router, const bool onInit=false, const bool withTaz=false, const bool silent=false, const MSEdge *sink=nullptr)=0
Performs a rerouting using the given router.
virtual SUMOTime sawBlockedParkingArea(const MSStoppingPlace *pa, bool local) const =0
virtual bool replaceRouteEdges(ConstMSEdgeVector &edges, double cost, double savings, const std::string &info, bool onInit=false, bool check=false, bool removeStops=true, std::string *msgReturn=nullptr)=0
Replaces the current route by the given edges.
virtual bool replaceParkingArea(MSParkingArea *parkingArea, std::string &errorMsg)=0
Replaces a stop.
virtual bool hasStops() const =0
Returns whether the vehicle has to stop somewhere.
virtual MSParkingArea * getNextParkingArea()=0
virtual void rememberParkingAreaScore(const MSStoppingPlace *pa, const std::string &score)=0
virtual void rememberBlockedParkingArea(const MSStoppingPlace *pa, bool local)=0
virtual int getNumberParkingReroutes() const =0
virtual bool isOnRoad() const =0
Returns the information whether the vehicle is on a road (is simulated)
virtual const std::vector< MSTransportable * > & getPersons() const =0
retrieve riding persons
virtual void replaceParameter(const SUMOVehicleParameter *newParameter)=0
Replaces the vehicle's parameter.
virtual void resetParkingAreaScores()=0
virtual void setArrivalPos(double arrivalPos)=0
Sets this vehicle's desired arrivalPos for its current route.
virtual const ConstMSEdgeVector::const_iterator & getCurrentRouteEdge() const =0
Returns an iterator pointing to the current edge in this vehicles route.
virtual MSStop & getNextStop()=0
virtual void setNumberParkingReroutes(int value)=0
virtual const MSRoute & getRoute() const =0
Returns the current route.
Structure representing possible vehicle parameter.
double arrivalPos
(optional) The position the vehicle shall arrive on
ArrivalPosDefinition arrivalPosProcedure
Information how the vehicle shall choose the arrival position.
A scoped lock which only triggers on condition.
std::vector< std::string > getVector()
return vector of strings
A wrapper for a Command function.
SUMOTime begin
The begin time these definitions are valid.
RandomDistributor< MSStoppingPlaceRerouter::StoppingPlaceVisible > parkProbs
The distributions of new parking areas to use as destinations.
MSEdgeVector siding
The list of siding edges.
MSEdgeVector main
The list of main edges (const and non-const for different usage)
RandomDistributor< ConstMSRoutePtr > routeProbs
The distributions of new routes to use.
SUMOTime end
The end time these definitions are valid.
RandomDistributor< MSEdge * > edgeProbs
The distributions of new destinations or vias to use.
bool isVia
The edge probs are vias and not destinations.
MSEdgeVector closedLanesAffected
The list of edges that are affect by closed lanes.
SVCPermissions permissions
The permissions to use.
MSEdgeVector closed
The list of closed edges.
std::vector< MSLane * > closedLanes
The list of closed lanes.