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 }
221 // @caution microsim uses current travel time teleport duration
222 const SUMOTime teleArrival = leaveTime + TIME2STEPS(veh->getEdge()->getLength() / MAX2(veh->getEdge()->getSpeedLimit(), NUMERICAL_EPS));
223 const bool atDest = veh->moveRoutePointer();
224 if (atDest) {
225 // teleporting to end of route
226 changeSegment(veh, teleArrival, nullptr, MSMoveReminder::NOTIFICATION_TELEPORT_ARRIVED, true);
227 } else {
228 veh->setEventTime(teleArrival);
229 addLeaderCar(veh, nullptr);
230 // teleporting vehicles must react to rerouters
233 }
234 }
235}
236
237
238void
240 myLeaderCars[veh->getEventTime()].push_back(veh);
241 veh->setApproaching(link);
242}
243
244
245void
247 myLeaderCars.clear();
248}
249
250
251bool
253 const auto candIt = myLeaderCars.find(v->getEventTime());
254 if (candIt != myLeaderCars.end()) {
255 std::vector<MEVehicle*>& cands = candIt->second;
256 auto it = find(cands.begin(), cands.end(), v);
257 if (it != cands.end()) {
258 cands.erase(it);
259 return true;
260 }
261 }
262 return false;
263}
264
265
266void
268 int qIdx = 0;
269 v->getSegment()->send(v, nullptr, qIdx, MSNet::getInstance()->getCurrentTimeStep(), reason);
271}
272
273
276 if (s != nullptr) { // vehicle is not teleporting
277 MESegment* next = s->getNextSegment();
278 if (next != nullptr) {
279 // ok, the street continues
280 return next;
281 }
282 }
283 // we have to check the next edge in the vehicle's route
284 const MSEdge* nextEdge = v->succEdge(1);
285 if (nextEdge == nullptr) {
286 // end of route
287 return nullptr;
288 }
289 return myEdges2FirstSegments[nextEdge->getNumericalID()];
290}
291
292
293int
294MELoop::numSegmentsFor(const double length, const double sLength) {
295 int no = (int)floor(length / sLength + 0.5);
296 if (no == 0) { // assure there is at least one segment
297 return 1;
298 } else {
299 return no;
300 }
301}
302
303
304void
307 const double length = e.getLength();
308 const int numSegments = numSegmentsFor(length, edgeType.edgeLength);
309 const double slength = length / (double)numSegments;
310 MESegment* newSegment = nullptr;
311 MESegment* nextSegment = nullptr;
312 const bool laneQueue = oc.getBool("meso-lane-queue");
313 bool multiQueue = laneQueue || (oc.getBool("meso-multi-queue") && e.getLanes().size() > 1 && e.getNumSuccessors() > 1);
314 for (int s = numSegments - 1; s >= 0; s--) {
315 std::string id = e.getID() + ":" + toString(s);
316 newSegment = new MESegment(id, e, nextSegment, slength, e.getLanes()[0]->getSpeedLimit(), s, multiQueue, edgeType);
317 multiQueue = laneQueue;
318 nextSegment = newSegment;
319 }
320 while (e.getNumericalID() >= static_cast<int>(myEdges2FirstSegments.size())) {
321 myEdges2FirstSegments.push_back(0);
322 }
323 myEdges2FirstSegments[e.getNumericalID()] = newSegment;
324 for (MSLane* lane : e.getLanes()) {
325 lane->updateMesoGUISegments();
326 }
327}
328
329
331MELoop::getSegmentForEdge(const MSEdge& e, double pos) {
332 if (e.getNumericalID() >= (int)myEdges2FirstSegments.size()) {
333 return nullptr;
334 }
336 if (pos > 0) {
337 double cpos = 0;
338 while (s->getNextSegment() != nullptr && cpos + s->getLength() < pos) {
339 cpos += s->getLength();
340 s = s->getNextSegment();
341 }
342 }
343 return s;
344}
345
346
347bool
349 for (const MSEdge* succ : e.getSuccessors()) {
350 if (succ->isRoundabout()) {
351 return true;
352 }
353 }
354 return false;
355}
356
357
358/****************************************************************************/
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:275
static bool isEnteringRoundabout(const MSEdge &e)
whether the given edge is entering a roundabout
Definition MELoop.cpp:348
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:331
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:252
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:267
void addLeaderCar(MEVehicle *veh, MSLink *link)
Adds the given car to the leading vehicles.
Definition MELoop.cpp:239
void clearState()
Remove all vehicles before quick-loading state.
Definition MELoop.cpp:246
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:294
void buildSegmentsFor(const MSEdge &e, const OptionsCont &oc)
Build the segments for a given edge.
Definition MELoop.cpp:305
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:284
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:255
void setEventTime(SUMOTime t, bool hasDelay=true)
Sets the (planned) time at which the vehicle leaves its current segment.
Definition MEVehicle.h:194
void setApproaching(MSLink *link)
registers vehicle with the given link
MESegment * getSegment() const
Returns the current segment the vehicle is on.
Definition MEVehicle.h:224
int getQueIndex() const
Returns the index of the que the vehicle is in.
Definition MEVehicle.h:233
virtual void setSegment(MESegment *s, int idx=0)
Sets the current segment the vehicle is at together with its que.
Definition MEVehicle.h:215
SUMOTime getEventTime() const
Returns the (planned) time at which the vehicle leaves its current segment.
Definition MEVehicle.h:206
void setBlockTime(const SUMOTime t)
Sets the time at which the vehicle was blocked.
Definition MEVehicle.h:269
SUMOTime getBlockTime() const
Returns the time at which the vehicle was blocked.
Definition MEVehicle.h:278
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
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
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 MSEdgeVector & getSuccessors(SUMOVehicleClass vClass=SVC_IGNORING) const
Returns the following edges, restricted by vClass.
Definition MSEdge.cpp:1286
static SUMOTime gTimeToTeleportDisconnected
Definition MSGlobals.h:66
static bool gCheckRoutes
Definition MSGlobals.h:91
static bool gRemoveGridlocked
Definition MSGlobals.h:75
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:187
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:417
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