Eclipse SUMO - Simulation of Urban MObility
Loading...
Searching...
No Matches
MELoop.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/****************************************************************************/
18// The main mesocopic simulation loop
19/****************************************************************************/
20#include <config.h>
21
22#include <queue>
23#include <vector>
24#include <map>
25#include <cmath>
26
27#include <microsim/MSNet.h>
28#include <microsim/MSEdge.h>
29#include <microsim/MSGlobals.h>
30#include <microsim/MSLane.h>
31#include <microsim/MSVehicle.h>
38#include "MELoop.h"
39#include "MESegment.h"
40#include "MEVehicle.h"
41
42
43// ===========================================================================
44// method definitions
45// ===========================================================================
46MELoop::MELoop(const SUMOTime recheckInterval) : myFullRecheckInterval(recheckInterval), myLinkRecheckInterval(TIME2STEPS(1)) {
47}
48
50 for (std::vector<MESegment*>::const_iterator j = myEdges2FirstSegments.begin(); j != myEdges2FirstSegments.end(); ++j) {
51 for (MESegment* s = *j; s != nullptr;) {
52 MESegment* n = s->getNextSegment();
53 delete s;
54 s = n;
55 }
56 }
57}
58
59
60void
62 while (!myLeaderCars.empty()) {
63 const SUMOTime time = myLeaderCars.begin()->first;
64 std::vector<MEVehicle*> vehs = myLeaderCars[time];
65 assert(time > tMax - DELTA_T || vehs.size() == 0);
66 if (time > tMax) {
67 return;
68 }
69 myLeaderCars.erase(time);
70 for (std::vector<MEVehicle*>::const_iterator i = vehs.begin(); i != vehs.end(); ++i) {
71 checkCar(*i);
72 assert(myLeaderCars.empty() || myLeaderCars.begin()->first >= time);
73 }
74 }
75}
76
77
79MELoop::changeSegment(MEVehicle* veh, SUMOTime leaveTime, MESegment* const toSegment, MSMoveReminder::Notification reason, const bool ignoreLink) const {
80 int qIdx = 0;
81 MESegment* const onSegment = veh->getSegment();
82 if (MESegment::isInvalid(toSegment)) {
83 if (veh->isStoppedTriggered()) {
84 return leaveTime + MAX2(SUMOTime(1), myLinkRecheckInterval);
85 }
86 if (onSegment != nullptr) {
87 onSegment->send(veh, toSegment, qIdx, leaveTime, reason);
88 } else {
89 WRITE_WARNINGF(TL("Vehicle '%' teleports beyond arrival edge '%', time=%."),
90 veh->getID(), veh->getEdge()->getID(), time2string(leaveTime));
91 }
92 veh->setSegment(toSegment); // signal arrival
94 return leaveTime;
95 } else if (!MSGlobals::gCheckRoutes && !ignoreLink && !MESegment::isInvalid(onSegment) && &onSegment->getEdge() != &toSegment->getEdge() &&
96 veh->getEdge()->allowedLanes(*veh->succEdge(1), veh->getVClass()) == nullptr) {
97 if (veh->isStopped()) {
98 veh->processStop();
99 }
100 return SUMOTime_MAX;
101 }
102 const SUMOTime entry = toSegment->hasSpaceFor(veh, leaveTime, qIdx);
103 if (entry == leaveTime && (ignoreLink || veh->mayProceed())) {
104 if (onSegment != nullptr) {
105 if (veh->getQueIndex() == MESegment::PARKING_QUEUE) { // parking or just aborted parking
106 if (veh->isParking()) {
107 veh->processStop();
108 }
109 veh->getEdge()->getLanes()[0]->removeParking(veh); // TODO for GUI only
110 } else {
111 onSegment->send(veh, toSegment, qIdx, leaveTime, onSegment->getNextSegment() == nullptr ? MSMoveReminder::NOTIFICATION_JUNCTION : MSMoveReminder::NOTIFICATION_SEGMENT);
112 }
113 toSegment->receive(veh, qIdx, leaveTime, false, ignoreLink, &onSegment->getEdge() != &toSegment->getEdge());
114 } else {
115 WRITE_WARNINGF(TL("Vehicle '%' ends teleporting on edge '%':%, time=%."),
116 veh->getID(), toSegment->getEdge().getID(), toSegment->getIndex(), time2string(leaveTime));
117 // this is not quite correct but suffices for interrogation by
118 // subsequent methods (veh->getSpeed() needs segment != 0)
120 // clean up detectors (do not add traffic data)
121 // note: updateDatector is not called if leaveTime == getLastEntryTime()
123 toSegment->receive(veh, qIdx, leaveTime, false, true, true);
124 }
125 return entry;
126 }
127 if (entry == leaveTime && !ignoreLink) { // this is a long way of saying !veh->mayProceed() (which is a costly call)
128 return entry + MAX2(SUMOTime(1), myLinkRecheckInterval);
129 }
130 return entry;
131}
132
133
134void
136 const SUMOTime leaveTime = veh->getEventTime();
137 MESegment* const onSegment = veh->getSegment();
138 MESegment* const toSegment = veh->getQueIndex() == MESegment::PARKING_QUEUE ? onSegment : nextSegment(onSegment, veh);
139 const bool teleporting = (onSegment == nullptr); // is the vehicle currently teleporting?
140 // @note reason is only evaluated if toSegment == nullptr
141 const SUMOTime nextEntry = changeSegment(veh, leaveTime, toSegment, MSMoveReminder::NOTIFICATION_ARRIVED, teleporting);
142 if (nextEntry == leaveTime) {
143 return;
144 }
147 if (!veh->isStopped() && (r1 || r3)) {
148 const bool disconnected = (MSGlobals::gTimeToTeleportDisconnected >= 0
149 && veh->succEdge(1) != nullptr
150 && veh->getEdge()->allowedLanes(*veh->succEdge(1), veh->getVClass()) == nullptr);
151 if ((r1 && !disconnected) || (r3 && disconnected)) {
152 teleportVehicle(veh, toSegment, disconnected);
153 return;
154 }
155 }
156 if (veh->getBlockTime() == SUMOTime_MAX && (!veh->isStopped()
157 || (!veh->isStoppedTriggered() && veh->isStoppedParking()))) {
158 veh->setBlockTime(leaveTime);
159 }
160 if (nextEntry == SUMOTime_MAX) {
161 // all usable queues on the next segment are full
162 SUMOTime newEventTime = MAX3(toSegment->getEventTime() + 1, leaveTime + 1, leaveTime + myFullRecheckInterval);
164 // if teleporting is enabled, make sure we look at the vehicle when the gridlock-time is up
166 newEventTime = MAX2(MIN2(newEventTime, veh->getBlockTime() + recheck + 1), leaveTime + DELTA_T);
167 }
168 veh->setEventTime(newEventTime);
169 } else {
170 // receiving segment has recently received another vehicle or the junction is blocked
171 veh->setEventTime(nextEntry);
172 }
173 addLeaderCar(veh, teleporting ? nullptr : onSegment->getLink(veh));
174}
175
176
177void
178MELoop::teleportVehicle(MEVehicle* veh, MESegment* const toSegment, bool disconnected) {
179 const SUMOTime leaveTime = veh->getEventTime();
180 MESegment* const onSegment = veh->getSegment();
182 WRITE_WARNINGF(TL("Teleporting vehicle '%'; waited too long, from edge '%':%, time=%."),
183 veh->getID(), onSegment->getEdge().getID(), onSegment->getIndex(),
184 time2string(leaveTime));
186 int qIdx = 0;
187 onSegment->send(veh, nullptr, qIdx, leaveTime, MSMoveReminder::NOTIFICATION_TELEPORT_ARRIVED);
188 veh->setSegment(nullptr);
190 return;
191 }
192 const bool teleporting = (onSegment == nullptr); // is the vehicle already teleporting?
193 // try to find a place on the current edge
194 MESegment* teleSegment = disconnected ? toSegment : toSegment->getNextSegment();
195 while (teleSegment != nullptr && changeSegment(veh, leaveTime, teleSegment, MSMoveReminder::NOTIFICATION_TELEPORT, true) != leaveTime) {
196 // @caution the time to get to the next segment here is ignored XXX
197 teleSegment = teleSegment->getNextSegment();
198 }
199 if (teleSegment != nullptr) {
200 if (!teleporting) {
201 // we managed to teleport in a single jump
202 const std::string reason = disconnected ? " (disconnected)" : "";
203 WRITE_WARNINGF(TL("Teleporting vehicle '%'; waited too long%, from edge '%':% to edge '%':%, time=%."),
204 veh->getID(), reason, onSegment->getEdge().getID(), onSegment->getIndex(),
205 teleSegment->getEdge().getID(), teleSegment->getIndex(), time2string(leaveTime));
207 }
208 } else {
209 // teleport across the current edge and try insertion later
210 if (!teleporting) {
211 int qIdx = 0;
212 // announce start of multi-step teleport, arrival will be announced in changeSegment()
213 WRITE_WARNINGF(TL("Teleporting vehicle '%'; waited too long, from edge '%':%, time=%."),
214 veh->getID(), onSegment->getEdge().getID(), onSegment->getIndex(), time2string(leaveTime));
216 // remove from current segment
217 onSegment->send(veh, nullptr, qIdx, leaveTime, MSMoveReminder::NOTIFICATION_TELEPORT);
218 // mark veh as teleporting
219 veh->setSegment(nullptr);
220 } else {
222 }
223 // @caution microsim uses current travel time teleport duration
224 const SUMOTime teleArrival = leaveTime + TIME2STEPS(veh->getEdge()->getLength() / MAX2(veh->getEdge()->getSpeedLimit(), NUMERICAL_EPS));
225 const bool atDest = veh->moveRoutePointer();
226 if (atDest) {
227 // teleporting to end of route
228 changeSegment(veh, teleArrival, nullptr, MSMoveReminder::NOTIFICATION_TELEPORT_ARRIVED, true);
229 } else {
230 veh->setEventTime(teleArrival);
231 addLeaderCar(veh, nullptr);
232 // teleporting vehicles must react to rerouters
235 }
236 }
237}
238
239
240void
242 myLeaderCars[veh->getEventTime()].push_back(veh);
243 veh->setApproaching(link);
244}
245
246
247void
249 myLeaderCars.clear();
250}
251
252
253bool
255 const auto candIt = myLeaderCars.find(v->getEventTime());
256 if (candIt != myLeaderCars.end()) {
257 std::vector<MEVehicle*>& cands = candIt->second;
258 auto it = find(cands.begin(), cands.end(), v);
259 if (it != cands.end()) {
260 cands.erase(it);
261 return true;
262 }
263 }
264 return false;
265}
266
267
268void
270 int qIdx = 0;
271 v->getSegment()->send(v, nullptr, qIdx, MSNet::getInstance()->getCurrentTimeStep(), reason);
273}
274
275
278 if (s != nullptr) { // vehicle is not teleporting
279 MESegment* next = s->getNextSegment();
280 if (next != nullptr) {
281 // ok, the street continues
282 return next;
283 }
284 }
285 // we have to check the next edge in the vehicle's route
286 const MSEdge* nextEdge = v->succEdge(1);
287 if (nextEdge == nullptr) {
288 // end of route
289 return nullptr;
290 }
291 if (MSGlobals::gUsingInternalLanes && s != nullptr && s->getEdge().isNormal()) {
292 const MSEdge* internal = s->getEdge().getInternalFollowingEdge(nextEdge, v->getVClass());
293 if (internal) {
294 nextEdge = internal;
295 }
296 }
297 return myEdges2FirstSegments[nextEdge->getNumericalID()];
298}
299
300
301int
302MELoop::numSegmentsFor(const double length, const double sLength) {
303 int no = (int)floor(length / sLength + 0.5);
304 if (no == 0) { // assure there is at least one segment
305 return 1;
306 } else {
307 return no;
308 }
309}
310
311
312void
315 const double length = e.getLength();
316 const int numSegments = numSegmentsFor(length, edgeType.edgeLength);
317 const double slength = length / (double)numSegments;
318 MESegment* newSegment = nullptr;
319 MESegment* nextSegment = nullptr;
320 const bool laneQueue = oc.getBool("meso-lane-queue");
321 bool multiQueue = laneQueue || (oc.getBool("meso-multi-queue") && e.getLanes().size() > 1 && e.getNumSuccessors() > 1);
322 for (int s = numSegments - 1; s >= 0; s--) {
323 std::string id = e.getID() + ":" + toString(s);
324 newSegment = new MESegment(id, e, nextSegment, slength, e.getLanes()[0]->getSpeedLimit(), s, multiQueue, edgeType);
325 multiQueue = laneQueue;
326 nextSegment = newSegment;
327 }
328 while (e.getNumericalID() >= static_cast<int>(myEdges2FirstSegments.size())) {
329 myEdges2FirstSegments.push_back(0);
330 }
331 myEdges2FirstSegments[e.getNumericalID()] = newSegment;
332 for (MSLane* lane : e.getLanes()) {
333 lane->updateMesoGUISegments();
334 }
335}
336
337
339MELoop::getSegmentForEdge(const MSEdge& e, double pos) {
340 if (e.getNumericalID() >= (int)myEdges2FirstSegments.size()) {
341 return nullptr;
342 }
344 if (pos > 0) {
345 double cpos = 0;
346 while (s->getNextSegment() != nullptr && cpos + s->getLength() < pos) {
347 cpos += s->getLength();
348 s = s->getNextSegment();
349 }
350 }
351 return s;
352}
353
354
355bool
357 for (const MSEdge* succ : e.getSuccessors()) {
358 if (succ->isRoundabout()) {
359 return true;
360 }
361 }
362 return false;
363}
364
365
366/****************************************************************************/
long long int SUMOTime
Definition GUI.h:36
#define WRITE_WARNINGF(...)
Definition MsgHandler.h:287
#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 TIME2STEPS(x)
Definition SUMOTime.h:60
T MIN2(T a, T b)
Definition StdDefs.h:80
T MAX2(T a, T b)
Definition StdDefs.h:86
T MAX3(T a, T b, T c)
Definition StdDefs.h:100
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:49
MESegment * nextSegment(MESegment *s, MEVehicle *v)
Retrieve next segment.
Definition MELoop.cpp:277
static bool isEnteringRoundabout(const MSEdge &e)
whether the given edge is entering a roundabout
Definition MELoop.cpp:356
MELoop(const SUMOTime recheckInterval)
SUMO constructor.
Definition MELoop.cpp:46
SUMOTime changeSegment(MEVehicle *veh, SUMOTime leaveTime, MESegment *const toSegment, MSMoveReminder::Notification reason, const bool ignoreLink=false) const
change to the next segment this handles combinations of the following cases: (ending / continuing rou...
Definition MELoop.cpp:79
MESegment * getSegmentForEdge(const MSEdge &e, double pos=0)
Get the segment for a given edge at a given position.
Definition MELoop.cpp:339
const SUMOTime myLinkRecheckInterval
the interval at which to recheck at blocked junctions (<=0 means asap)
Definition MELoop.h:149
void simulate(SUMOTime tMax)
Perform simulation up to the given time.
Definition MELoop.cpp:61
bool removeLeaderCar(MEVehicle *v)
Removes the given car from the leading vehicles.
Definition MELoop.cpp:254
const SUMOTime myFullRecheckInterval
the interval at which to recheck at full segments (<=0 means asap)
Definition MELoop.h:146
void vaporizeCar(MEVehicle *v, MSMoveReminder::Notification reason)
remove the given car and clean up the relevant data structures
Definition MELoop.cpp:269
void addLeaderCar(MEVehicle *veh, MSLink *link)
Adds the given car to the leading vehicles.
Definition MELoop.cpp:241
void clearState()
Remove all vehicles before quick-loading state.
Definition MELoop.cpp:248
std::vector< MESegment * > myEdges2FirstSegments
mapping from internal edge ids to their initial segments
Definition MELoop.h:143
std::map< SUMOTime, std::vector< MEVehicle * > > myLeaderCars
leader cars in the segments sorted by exit time
Definition MELoop.h:140
~MELoop()
Definition MELoop.cpp:49
void checkCar(MEVehicle *veh)
Check whether the vehicle may move.
Definition MELoop.cpp:135
void teleportVehicle(MEVehicle *veh, MESegment *const toSegment, bool disconnected)
teleports a vehicle or continues a teleport
Definition MELoop.cpp:178
static int numSegmentsFor(const double length, const double slength)
Compute number of segments per edge (best value stay close to the configured segment length)
Definition MELoop.cpp:302
void buildSegmentsFor(const MSEdge &e, const OptionsCont &oc)
Build the segments for a given edge.
Definition MELoop.cpp:313
A single mesoscopic segment (cell)
Definition MESegment.h:50
void addReminders(MEVehicle *veh) const
add this lanes MoveReminders to the given vehicle
static const int PARKING_QUEUE
Definition MESegment.h:53
void receive(MEVehicle *veh, const int qIdx, SUMOTime time, const bool isDepart=false, const bool isTeleport=false, const bool newEdge=false)
Adds the vehicle to the segment, adapting its parameters.
MSLink * getLink(const MEVehicle *veh, bool tlsPenalty=false) const
Returns the link the given car will use when passing the next junction.
double getLength() const
Returns the length of the segment in meters.
Definition MESegment.h:246
SUMOTime hasSpaceFor(const MEVehicle *const veh, const SUMOTime entryTime, int &qIdx, const bool init=false) const
Returns whether the given vehicle would still fit into the segment.
const MSEdge & getEdge() const
Returns the edge this segment belongs to.
Definition MESegment.h:366
void send(MEVehicle *veh, MESegment *const next, const int nextQIdx, SUMOTime time, const MSMoveReminder::Notification reason)
Removes the vehicle from the segment, adapting its parameters.
SUMOTime getEventTime() const
Returns the (planned) time at which the next vehicle leaves this segment.
int getIndex() const
Returns the running index of the segment in the edge (0 is the most upstream).
Definition MESegment.h:230
static bool isInvalid(const MESegment *segment)
whether the given segment is 0 or encodes vaporization
Definition MESegment.h:449
MESegment * getNextSegment() const
Returns the following segment on the same edge (0 if it is the last).
Definition MESegment.h:238
A vehicle from the mesoscopic point of view.
Definition MEVehicle.h:42
bool mayProceed()
Returns whether the vehicle is allowed to pass the next junction, checks also for triggered stops.
void processStop()
ends the current stop and performs loading/unloading
SUMOTime getWaitingTime(const bool accumulated=false) const
Returns the duration for which the vehicle was blocked.
Definition MEVehicle.h:289
bool moveRoutePointer()
Update when the vehicle enters a new edge in the move step.
void updateDetectors(const SUMOTime currentTime, const SUMOTime exitTime, const bool isLeave, const MSMoveReminder::Notification reason=MSMoveReminder::NOTIFICATION_JUNCTION)
Updates all vehicle detectors.
SUMOTime getLastEntryTime() const
Returns the time the vehicle entered the current segment.
Definition MEVehicle.h:260
void setEventTime(SUMOTime t, bool hasDelay=true)
Sets the (planned) time at which the vehicle leaves its current segment.
Definition MEVehicle.h:199
void setApproaching(MSLink *link)
registers vehicle with the given link
MESegment * getSegment() const
Returns the current segment the vehicle is on.
Definition MEVehicle.h:229
int getQueIndex() const
Returns the index of the que the vehicle is in.
Definition MEVehicle.h:238
virtual void setSegment(MESegment *s, int idx=0)
Sets the current segment the vehicle is at together with its que.
Definition MEVehicle.h:220
SUMOTime getEventTime() const
Returns the (planned) time at which the vehicle leaves its current segment.
Definition MEVehicle.h:211
void setBlockTime(const SUMOTime t)
Sets the time at which the vehicle was blocked.
Definition MEVehicle.h:274
SUMOTime getBlockTime() const
Returns the time at which the vehicle was blocked.
Definition MEVehicle.h:283
const MSEdge * succEdge(int nSuccs) const
Returns the nSuccs'th successor of edge the vehicle is currently at.
bool isStoppedParking() const
Returns whether the vehicle is on a parking stop.
bool isParking() const
Returns whether the vehicle is parking.
const MSEdge * getEdge() const
Returns the edge the vehicle is currently at.
virtual void activateReminders(const MSMoveReminder::Notification reason, const MSLane *enteredLane=0)
"Activates" all current move reminder
SUMOVehicleClass getVClass() const
Returns the vehicle's access class.
bool isStoppedTriggered() const
Returns whether the vehicle is on a triggered stop.
bool isStopped() const
Returns whether the vehicle is at a stop.
A road/street connecting two junctions.
Definition MSEdge.h:77
int getNumSuccessors() const
Returns the number of edges that may be reached from this edge.
Definition MSEdge.h:388
const std::vector< MSLane * > & getLanes() const
Returns this edge's lanes.
Definition MSEdge.h:168
const std::vector< MSLane * > * allowedLanes(const MSEdge &destination, SUMOVehicleClass vclass=SVC_IGNORING, bool ignoreTransientPermissions=false) const
Get the allowed lanes to reach the destination-edge.
Definition MSEdge.cpp:480
bool isNormal() const
return whether this edge is an internal edge
Definition MSEdge.h:264
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:1190
double getLength() const
return the length of the edge
Definition MSEdge.h:694
int getNumericalID() const
Returns the numerical id of the edge.
Definition MSEdge.h:307
const std::string & getEdgeType() const
Returns the type of the edge.
Definition MSEdge.h:320
const MSEdge * getInternalFollowingEdge(const MSEdge *followerAfterInternal, SUMOVehicleClass vClass) const
Definition MSEdge.cpp:932
const MSEdgeVector & getSuccessors(SUMOVehicleClass vClass=SVC_IGNORING) const
Returns the following edges, restricted by vClass.
Definition MSEdge.cpp:1290
static SUMOTime gTimeToTeleportDisconnected
Definition MSGlobals.h:66
static bool gCheckRoutes
Definition MSGlobals.h:91
static bool gRemoveGridlocked
Definition MSGlobals.h:75
static bool gUsingInternalLanes
Information whether the simulation regards internal lanes.
Definition MSGlobals.h:81
static SUMOTime gTimeToGridlock
Definition MSGlobals.h:57
Representation of a lane in the micro simulation.
Definition MSLane.h:84
Notification
Definition of a vehicle state.
@ NOTIFICATION_ARRIVED
The vehicle arrived at its destination (is deleted)
@ NOTIFICATION_TELEPORT_ARRIVED
The vehicle was teleported out of the net.
@ NOTIFICATION_SEGMENT
The vehicle changes the segment (meso only)
@ NOTIFICATION_JUNCTION
The vehicle arrived at a junction.
@ NOTIFICATION_TELEPORT
The vehicle is being teleported.
static MSNet * getInstance()
Returns the pointer to the unique instance of MSNet (singleton).
Definition MSNet.cpp:199
MSVehicleControl & getVehicleControl()
Returns the vehicle control.
Definition MSNet.h:392
const MESegment::MesoEdgeType & getMesoType(const std::string &typeID)
Returns edge type specific meso parameters if no type specific parameters have been loaded,...
Definition MSNet.cpp:426
void registerTeleportJam()
register one non-collision-related teleport
void scheduleVehicleRemoval(SUMOVehicle *veh, bool checkDuplicate=false)
Removes a vehicle after it has ended.
const std::string & getID() const
Returns the id.
Definition Named.h:74
A storage for options typed value containers)
Definition OptionsCont.h:89
bool getBool(const std::string &name) const
Returns the boolean-value of the named option (only for Option_Bool)
edge type specific meso parameters
Definition MESegment.h:58