Eclipse SUMO - Simulation of Urban MObility
Loading...
Searching...
No Matches
MSTransportableControl.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/****************************************************************************/
21// Stores all persons in the net and handles their waiting for cars.
22/****************************************************************************/
23#include <config.h>
24
25#include <vector>
26#include <algorithm>
33#ifdef JPS_VERSION
35#endif
40#include <microsim/MSNet.h>
41#include <microsim/MSEdge.h>
42#include <microsim/MSVehicle.h>
44
45
46// ===========================================================================
47// method definitions
48// ===========================================================================
50 myLoadedNumber(0),
51 myDiscardedNumber(0),
52 myRunningNumber(0),
53 myJammedNumber(0),
54 myWaitingForDepartureNumber(0),
55 myWaitingForVehicleNumber(0),
56 myWaitingUntilNumber(0),
57 myAccessNumber(0),
58 myEndedNumber(0),
59 myArrivedNumber(0),
60 myTeleportsAbortWait(0),
61 myTeleportsWrongDest(0),
62 myHaveNewWaiting(false) {
64 MSNet* const net = MSNet::getInstance();
66 if (isPerson) {
67 const std::string& model = oc.getString("pedestrian.model");
68 if (model == "striping") {
70#ifdef JPS_VERSION
71 } else if (model == "jupedsim") {
73#endif
74 } else if (model != "nonInteracting") {
76 throw ProcessError(TLF("Unknown pedestrian model '%'", model));
77 }
78 }
79 if (oc.isSet("vehroute-output")) {
81 }
82 if (oc.isSet("personroute-output")) {
83 OutputDevice::createDeviceByOption("personroute-output", "routes", "routes_file.xsd");
85 }
86 if (oc.isSet("personinfo-output")) {
87 OutputDevice::createDeviceByOption("personinfo-output", "tripinfos", "tripinfo_file.xsd");
88 }
89 myAbortWaitingTimeout = string2time(oc.getString("time-to-teleport.ride"));
90 myMaxTransportableNumber = isPerson ? oc.getInt("max-num-persons") : -1;
91}
92
93
101
102
103bool
105 const SUMOVehicleParameter& param = transportable->getParameter();
106 if (myTransportables.find(param.id) == myTransportables.end()) {
107 myTransportables[param.id] = transportable;
108 const SUMOTime step = param.depart % DELTA_T == 0 ? param.depart : (param.depart / DELTA_T + 1) * DELTA_T;
109 myWaiting4Departure[step].push_back(transportable);
112 return true;
113 }
114 return false;
115}
116
117
118void
121 if (transportable->hasDeparted()) {
122 const SUMOVehicleParameter& param = transportable->getParameter();
123 const SUMOTime step = param.depart % DELTA_T == 0 ? param.depart : (param.depart / DELTA_T + 1) * DELTA_T;
125 auto it = std::find(waiting.begin(), waiting.end(), transportable);
126 if (it != waiting.end()) {
127 waiting.erase(it);
128 if (waiting.size() == 0) {
129 myWaiting4Departure.erase(step);
130 }
131 }
132 }
133}
134
135
137MSTransportableControl::get(const std::string& id) const {
138 std::map<std::string, MSTransportable*>::const_iterator i = myTransportables.find(id);
139 if (i == myTransportables.end()) {
140 return nullptr;
141 }
142 return (*i).second;
143}
144
145
146void
149 if (oc.isSet("personinfo-output")) {
150 transportable->tripInfoOutput(OutputDevice::getDeviceByOption("personinfo-output"));
151 } else if (oc.isSet("tripinfo-output")) {
152 transportable->tripInfoOutput(OutputDevice::getDeviceByOption("tripinfo-output"));
153 } else if (oc.getBool("duration-log.statistics")) {
154 // collecting statistics is a sideffect
156 transportable->tripInfoOutput(dev);
157 }
158 if (oc.isSet("vehroute-output") || oc.isSet("personroute-output")) {
159 if (transportable->hasArrived() || oc.getBool("vehroute-output.write-unfinished")) {
160 // @todo: adapt this if we ever introduce person-device.vehroute
161 if (transportable->getBoolParam("has.vehroute.person-device", false, true)) {
162 if (oc.getBool("vehroute-output.sorted")) {
163 const SUMOTime departure = oc.getBool("vehroute-output.intended-depart") ? transportable->getParameter().depart : transportable->getDeparture();
165 transportable->routeOutput(od, oc.getBool("vehroute-output.route-length"));
167 departure, transportable->getID(), od.getString());
168 } else {
169 transportable->routeOutput(*myRouteInfos.routeOut, oc.getBool("vehroute-output.route-length"));
170 }
171 }
172 }
173 }
174 const std::map<std::string, MSTransportable*>::iterator i = myTransportables.find(transportable->getID());
175 if (i != myTransportables.end()) {
176 if (i->second->hasDeparted()) {
178 }
182 delete i->second;
183 myTransportables.erase(i);
184 }
185}
186
187
188void
190 // avoid taxi device having dangling references to erased customers
192 while (loadedBegin() != loadedEnd()) {
193 erase(loadedBegin()->second);
194 }
195}
196
197
198void
200 const SUMOTime step = time % DELTA_T == 0 ? time : (time / DELTA_T + 1) * DELTA_T;
201 // avoid double registration
202 const TransportableVector& transportables = myWaitingUntil[step];
203 if (std::find(transportables.begin(), transportables.end(), transportable) == transportables.end()) {
204 myWaitingUntil[step].push_back(transportable);
206 }
207}
208
209
210void
212 myHaveNewWaiting = false;
213 while (myWaiting4Departure.find(time) != myWaiting4Departure.end()) {
214 TransportableVector& transportables = myWaiting4Departure[time];
215 // we cannot use an iterator here because there might be additions to the vector while proceeding
216 for (auto it = transportables.begin(); it != transportables.end();) {
217 MSTransportable* t = *it;
220 nextStep.insert(nextStep.begin(), transportables.begin(), transportables.end());
221 transportables.clear();
222 break;
223 } else if (myMaxTransportableNumber == 0) {
224 erase(t);
225 it = transportables.erase(it);
226 continue;
227 }
228 it = transportables.erase(it);
230 const bool isPerson = t->isPerson();
231 t->setDeparted(time);
232 if (t->proceed(net, time)) {
237 if (oc.getBool("vehroute-output.sorted")) {
238 const SUMOTime departure = oc.getBool("vehroute-output.intended-depart") ? t->getParameter().depart : time;
239 if (oc.isSet("personroute-output")) {
240 myRouteInfos.departureCounts[departure]++;
241 } else {
243 }
244 }
245 } else {
246 erase(t);
247 }
248 }
249 myWaiting4Departure.erase(time);
250 }
251 while (myWaitingUntil.find(time) != myWaitingUntil.end()) {
252 // make a copy because 0-duration stops might modify the vector
253 const TransportableVector transportables = myWaitingUntil[time];
254 myWaitingUntil.erase(time);
255 for (MSTransportable* t : transportables) {
257 if (!t->proceed(net, time)) {
258 erase(t);
259 }
260 }
261 }
262}
263
264
265void
269
270
271void
273 myWaiting4Vehicle[edge].push_back(transportable);
275 myHaveNewWaiting = true;
276 if (myAbortWaitingTimeout >= 0) {
278 }
279}
280
281
282bool
284 const auto wait = myWaiting4Vehicle.find(edge);
285 if (wait != myWaiting4Vehicle.end()) {
286 for (const MSTransportable* t : wait->second) {
287 if (t->isWaitingFor(vehicle)
288 && vehicle->allowsBoarding(t)
289 && vehicle->isStoppedInRange(t->getEdgePos(), MSGlobals::gStopTolerance, true)) {
290 return true;
291 }
292 }
293 }
294 return false;
295}
296
297
298bool
299MSTransportableControl::loadAnyWaiting(const MSEdge* edge, SUMOVehicle* vehicle, SUMOTime& timeToLoadNext, SUMOTime& stopDuration, MSTransportable* const force) {
300 bool ret = false;
301 const auto wait = myWaiting4Vehicle.find(edge);
302 if (wait != myWaiting4Vehicle.end()) {
303 const SUMOTime currentTime = SIMSTEP;
304 TransportableVector& transportables = wait->second;
305 for (TransportableVector::iterator i = transportables.begin(); i != transportables.end();) {
306 MSTransportable* const t = *i;
307 if (t->isWaitingFor(vehicle) && (t == force ||
308 (vehicle->allowsBoarding(t)
309 && timeToLoadNext - DELTA_T <= currentTime
311 edge->removeTransportable(t);
312 vehicle->addTransportable(t);
313 if (myAbortWaitingTimeout >= 0) {
314 t->setAbortWaiting(-1);
315 }
316 if (timeToLoadNext >= 0) { // meso does not have loading times
317 const SUMOTime loadingDuration = (SUMOTime)((double)vehicle->getVehicleType().getLoadingDuration(t->isPerson()) * t->getVehicleType().getBoardingFactor());
318 //update the time point at which the next transportable can be loaded on the vehicle
319 if (timeToLoadNext > currentTime - DELTA_T) {
320 timeToLoadNext += loadingDuration;
321 } else {
322 timeToLoadNext = currentTime + loadingDuration;
323 }
324 }
325
326 static_cast<MSStageDriving*>(t->getCurrentStage())->setVehicle(vehicle);
327 if (t->getCurrentStage()->getOriginStop() != nullptr) {
329 }
330 i = transportables.erase(i);
332 ret = true;
333 } else {
334 ++i;
335 }
336 }
337 if (transportables.empty()) {
338 myWaiting4Vehicle.erase(wait);
339 }
340 if (ret && timeToLoadNext >= 0) {
341 //if the time a transportable needs to get loaded on the vehicle extends the duration of the stop of the vehicle extend
342 //the duration by setting it to the loading duration of the transportable
343 stopDuration = MAX2(stopDuration, timeToLoadNext - currentTime);
344 }
345 }
346 return ret;
347}
348
349
350bool
354
355
356bool
360
361
362int
366
367
368int
372
373
374int
378
379int
383
384void
386 for (const auto& it : myWaiting4Vehicle) {
387 const MSEdge* edge = it.first;
388 for (MSTransportable* const p : it.second) {
389 edge->removeTransportable(p);
390 MSStageDriving* stage = dynamic_cast<MSStageDriving*>(p->getCurrentStage());
391 const std::string waitDescription = stage == nullptr ? "waiting" : stage->getWaitingDescription();
392 WRITE_WARNING(p->getObjectType() + " '" + p->getID() + "' aborted " + waitDescription + ".");
393 if (myAbortWaitingTimeout >= 0) {
394 p->setAbortWaiting(-1);
395 }
396 erase(p);
397 }
398 }
399 myWaiting4Vehicle.clear();
401}
402
403void
405 const MSEdge* edge = t->getEdge();
406 auto it = myWaiting4Vehicle.find(edge);
407 if (it != myWaiting4Vehicle.end()) {
408 TransportableVector& waiting = it->second;
409 auto it2 = std::find(waiting.begin(), waiting.end(), t);
410 if (it2 != waiting.end()) {
411 if (myAbortWaitingTimeout >= 0) {
412 (*it2)->setAbortWaiting(-1);
413 }
414 waiting.erase(it2);
416 }
417 }
418}
419
420void
422 for (std::map<SUMOTime, TransportableVector>::iterator it = myWaiting4Departure.begin(); it != myWaiting4Departure.end(); ++it) {
423 TransportableVector& ts = it->second;
424 TransportableVector::iterator it2 = std::find(ts.begin(), ts.end(), t);
425 if (it2 != ts.end()) {
426 ts.erase(it2);
427 }
428 }
429 for (std::map<SUMOTime, TransportableVector>::iterator it = myWaitingUntil.begin(); it != myWaitingUntil.end(); ++it) {
430 TransportableVector& ts = it->second;
431 TransportableVector::iterator it2 = std::find(ts.begin(), ts.end(), t);
432 if (it2 != ts.end()) {
433 ts.erase(it2);
434 }
435 }
436}
437
438
441 SumoRNG* rng) const {
442 const double speedFactor = vtype->computeChosenSpeedDeviation(rng);
443 return new MSPerson(pars, vtype, plan, speedFactor);
444}
445
446
449 return new MSTransportable(pars, vtype, plan, false);
450}
451
452
453void
455 std::ostringstream oss;
456 oss << myRunningNumber << " " << myLoadedNumber << " " << myEndedNumber << " " << myWaitingForDepartureNumber << " " << myArrivedNumber << " " << myDiscardedNumber;
457 oss << " " << myJammedNumber << " " << myWaitingForVehicleNumber << " " << myWaitingUntilNumber << " " << myHaveNewWaiting;
458 out.writeAttr(SUMO_ATTR_STATE, oss.str());
459 for (const auto& it : myTransportables) {
460 it.second->saveState(out);
461 }
462}
463
464
465void
471
472void
474 for (std::map<std::string, MSTransportable*>::iterator i = myTransportables.begin(); i != myTransportables.end(); ++i) {
475 delete (*i).second;
476 }
477 myTransportables.clear();
478 myWaiting4Vehicle.clear();
479 myWaiting4Departure.clear();
480 myWaitingUntil.clear();
481 myLoadedNumber = 0;
483 myRunningNumber = 0;
484 myJammedNumber = 0;
488 myEndedNumber = 0;
489 myArrivedNumber = 0;
490 myHaveNewWaiting = false;
493 }
495}
496
497/****************************************************************************/
long long int SUMOTime
Definition GUI.h:36
#define WRITE_WARNING(msg)
Definition MsgHandler.h:286
#define TLF(string,...)
Definition MsgHandler.h:306
SUMOTime DELTA_T
Definition SUMOTime.cpp:38
SUMOTime string2time(const std::string &r)
convert string to SUMOTime
Definition SUMOTime.cpp:46
#define SIMSTEP
Definition SUMOTime.h:64
@ SUMO_ATTR_STATE
The state of a link.
T MAX2(T a, T b)
Definition StdDefs.h:86
static void allCustomersErased()
signal the end of the simulation and the removal of all customers
static void writeSortedOutput(SortedRouteInfo *routeInfo, SUMOTime depart, const std::string &id, const std::string &xmlOutput)
static void registerTransportableDepart(SUMOTime depart)
A road/street connecting two junctions.
Definition MSEdge.h:77
virtual void removeTransportable(MSTransportable *t) const
Definition MSEdge.cpp:1225
static double gStopTolerance
The tolerance to apply when matching waiting persons and vehicles.
Definition MSGlobals.h:168
The simulated network and simulation perfomer.
Definition MSNet.h:89
void informTransportableStateListener(const MSTransportable *const transportable, TransportableState to, const std::string &info="")
Informs all added listeners about a transportable's state change.
Definition MSNet.cpp:1375
static MSNet * getInstance()
Returns the pointer to the unique instance of MSNet (singleton).
Definition MSNet.cpp:187
@ CONTAINER_DEPARTED
The transportable container has departed (was inserted into the network)
@ PERSON_DEPARTED
The transportable person has departed (was inserted into the network)
@ PERSON_ARRIVED
The transportable person arrived at his destination (is deleted)
@ CONTAINER_ARRIVED
The transportable container arrived at his destination (is deleted)
A pedestrian following model that acts as a proxy for pedestrian models provided by the JuPedSim thir...
The pedestrian following model.
The pedestrian movement model using stripes on sidewalks.
virtual int getActiveNumber()=0
return the number of active objects
virtual void clearState()=0
Resets pedestrians when quick-loading state.
std::string getWaitingDescription() const
Return where the person waits and for what.
virtual MSStoppingPlace * getOriginStop() const
returns the origin stop (if any). only needed for MSStageTrip
Definition MSStage.h:93
void removeTransportable(const MSTransportable *p)
Removes a transportable from this stop.
void loadState(const std::string &state)
Reconstruct the current state.
void setWaitEnd(SUMOTime time, MSTransportable *transportable)
sets the arrival time for a waiting transportable
bool hasTransportables() const
checks whether any transportable waits to finish her plan
int myAccessNumber
The number of transportables currently in an access stage.
constVehIt loadedBegin() const
Returns the begin of the internal transportables map.
int myWaitingForDepartureNumber
The number of transportables waiting for departure.
void forceDeparture()
register forced (traci) departure
void fixLoadCount(const MSTransportable *transportable)
decrement counter to avoid double counting transportables loaded from state
void checkWaiting(MSNet *net, const SUMOTime time)
checks whether any transportables waiting time is over
void saveState(OutputDevice &out)
Saves the current state into the given stream.
MSTransportableControl(const bool isPerson)
Constructor.
bool hasAnyWaiting(const MSEdge *edge, SUMOVehicle *vehicle) const
check whether any transportables are waiting for the given vehicle
void eraseAll()
@removes all transportables
int myLoadedNumber
The number of build transportables.
constVehIt loadedEnd() const
Returns the end of the internal transportables map.
std::map< std::string, MSTransportable * > myTransportables
all currently created transportables by id
void abortWaiting(MSTransportable *t)
aborts waiting stage of transportable
void abortWaitingForVehicle(MSTransportable *t)
let the given transportable abort waiting for a vehicle (when removing stage via TraCI)
void abortAnyWaitingForVehicle()
aborts the plan for any transportable that is still waiting for a ride
bool hasNonWaiting() const
checks whether any transportable is still engaged in walking / stopping
virtual MSTransportable * buildContainer(const SUMOVehicleParameter *pars, MSVehicleType *vtype, MSTransportable::MSTransportablePlan *plan) const
Builds a new container.
int myArrivedNumber
The number of transportables that arrived at their destination.
virtual MSTransportable * buildPerson(const SUMOVehicleParameter *pars, MSVehicleType *vtype, MSTransportable::MSTransportablePlan *plan, SumoRNG *rng) const
Builds a new person.
std::map< const MSEdge *, TransportableVector, ComparatorNumericalIdLess > myWaiting4Vehicle
the lists of waiting transportables
int getMovingNumber() const
Returns the number of transportables moving by themselvs (i.e. walking)
void addWaiting(const MSEdge *edge, MSTransportable *person)
adds a transportable to the list of transportables waiting for a vehicle on the specified edge
SUMOTime myAbortWaitingTimeout
The time until waiting for a ride is aborted.
int myEndedNumber
The number of transportables that exited the simulation.
int myWaitingUntilNumber
The number of transportables waiting for a specified time.
void clearState()
Resets transportables when quick-loading state.
virtual void erase(MSTransportable *transportable)
removes a single transportable
int myDiscardedNumber
The number of discarded transportables.
std::map< SUMOTime, TransportableVector > myWaitingUntil
the lists of walking / stopping transportables
bool add(MSTransportable *transportable)
Adds a single transportable, returns false if an id clash occurred.
bool myHaveNewWaiting
whether a new transportable waiting for a vehicle has been added in the last step
virtual ~MSTransportableControl()
Destructor.
int myJammedNumber
The number of jammed transportables.
int myMaxTransportableNumber
maximum transportable count
std::map< SUMOTime, TransportableVector > myWaiting4Departure
Transportables waiting for departure.
bool loadAnyWaiting(const MSEdge *edge, SUMOVehicle *vehicle, SUMOTime &timeToLoadNext, SUMOTime &stopDuration, MSTransportable *const force=nullptr)
load any applicable transportables Loads any person / container that is waiting on that edge for the ...
std::vector< MSTransportable * > TransportableVector
Definition of a list of transportables.
int myRunningNumber
The number of transportables within the network (build and inserted but not removed)
int myWaitingForVehicleNumber
The number of transportables waiting for vehicles.
MSTransportable * get(const std::string &id) const
Returns the named transportable, if existing.
MSDevice_Vehroutes::SortedRouteInfo myRouteInfos
Information needed to sort transportable output by departure time.
int getActiveCount()
return the number of active transportable objects
int getRidingNumber() const
Returns the number of transportables riding a vehicle.
virtual double getEdgePos() const
Return the position on the edge.
bool hasDeparted() const
return whether the transportable has started its plan
SUMOTime getDeparture() const
logs depart time of the current stage
void setAbortWaiting(const SUMOTime timeout)
void routeOutput(OutputDevice &os, const bool withRouteLength) const
Called on writing vehroute output.
MSStage * getCurrentStage() const
Return the current stage.
void setDeparted(SUMOTime now)
logs depart time of the current stage
virtual bool proceed(MSNet *net, SUMOTime time, const bool vehicleArrived=false)
bool isPerson() const override
Whether it is a person.
const SUMOVehicleParameter & getParameter() const override
Returns the vehicle's parameter (including departure definition)
const MSVehicleType & getVehicleType() const override
Returns the object's "vehicle" type.
void tripInfoOutput(OutputDevice &os) const
Called on writing tripinfo output.
const MSEdge * getEdge() const override
Returns the current edge.
std::vector< MSStage * > MSTransportablePlan
the structure holding the plan of a transportable
bool isWaitingFor(const SUMOVehicle *vehicle) const
Whether the transportable waits for the given vehicle in the current step.
bool hasArrived() const override
return whether the person has reached the end of its plan
The car-following model and parameter.
SUMOTime getLoadingDuration(const bool isPerson) const
Get this vehicle type's loading duration.
double getBoardingFactor() const
Get this person type's factor for loading/boarding duration.
double computeChosenSpeedDeviation(SumoRNG *rng, const double minDev=-1.) const
Computes and returns the speed deviation.
const std::string & getID() const
Returns the id.
Definition Named.h:74
A storage for options typed value containers)
Definition OptionsCont.h:89
bool isSet(const std::string &name, bool failOnNonExistant=true) const
Returns the information whether the named option is set.
int getInt(const std::string &name) const
Returns the int-value of the named option (only for Option_Integer)
std::string getString(const std::string &name) const
Returns the string-value of the named option (only for Option_String)
bool getBool(const std::string &name) const
Returns the boolean-value of the named option (only for Option_Bool)
static OptionsCont & getOptions()
Retrieves the options.
An output device that encapsulates an ofstream.
std::string getString() const
Returns the current content as a string.
Static storage of an output device and its base (abstract) implementation.
OutputDevice & writeAttr(const SumoXMLAttr attr, const T &val)
writes a named attribute
static bool createDeviceByOption(const std::string &optionName, const std::string &rootElement="", const std::string &schemaFile="")
Creates the device using the output definition stored in the named option.
static OutputDevice & getDeviceByOption(const std::string &name)
Returns the device described by the option.
bool getBoolParam(const std::string &paramName, const bool required=false, const bool deflt=false) const
Retrieve a boolean parameter for the traffic object.
virtual const MSVehicleType & getVehicleType() const =0
Returns the object's "vehicle" type.
Representation of a vehicle.
Definition SUMOVehicle.h:63
virtual bool isStoppedInRange(const double pos, const double tolerance, bool checkFuture=false) const =0
Returns whether the vehicle is stopped in the range of the given position.
virtual void addTransportable(MSTransportable *transportable)=0
Adds a person or container to this vehicle.
virtual bool allowsBoarding(const MSTransportable *t) const =0
whether the given transportable is allowed to board this vehicle
Structure representing possible vehicle parameter.
std::string id
The vehicle's id.
std::map< const SUMOTime, int > departureCounts
Map needed to sort vehicles by departure time.
OutputDevice * routeOut
route output device