Eclipse SUMO - Simulation of Urban MObility
Loading...
Searching...
No Matches
MSDevice_StationFinder.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/****************************************************************************/
19// A device which triggers rerouting to nearby charging stations
20/****************************************************************************/
21#include <config.h>
22
23#include <microsim/MSEdge.h>
26#include <microsim/MSNet.h>
28#include <microsim/MSStop.h>
39#include "MSRoutingEngine.h"
40#include "MSDevice_Battery.h"
42
43//#define DEBUG_STATIONFINDER_RESCUE
44//#define DEBUG_STATIONFINDER_REROUTE
45#define DEBUG_COND (myHolder.isSelected())
46
47
48// ===========================================================================
49// static variables
50// ===========================================================================
51
52
53// ===========================================================================
54// method definitions
55// ===========================================================================
56// ---------------------------------------------------------------------------
57// static initialisation methods
58// ---------------------------------------------------------------------------
59void
61 insertDefaultAssignmentOptions("stationfinder", "Battery", oc);
62 oc.doRegister("device.stationfinder.rescueTime", new Option_String("1800", "TIME"));
63 oc.addDescription("device.stationfinder.rescueTime", "Battery", TL("Time to wait for a rescue vehicle on the road side when the battery is empty"));
64 oc.doRegister("device.stationfinder.rescueAction", new Option_String("remove"));
65 oc.addDescription("device.stationfinder.rescueAction", "Battery", TL("How to deal with a vehicle which has to stop due to low battery: [none, remove, tow]"));
66 oc.doRegister("device.stationfinder.reserveFactor", new Option_Float(1.1));
67 oc.addDescription("device.stationfinder.reserveFactor", "Battery", TL("Scale battery need with this factor to account for unexpected traffic situations"));
68 oc.doRegister("device.stationfinder.emptyThreshold", new Option_Float(0.05));
69 oc.addDescription("device.stationfinder.emptyThreshold", "Battery", TL("Battery percentage to go into rescue mode"));
70 oc.doRegister("device.stationfinder.radius", new Option_String("180", "TIME"));
71 oc.addDescription("device.stationfinder.radius", "Battery", TL("Search radius in travel time seconds"));
72 oc.doRegister("device.stationfinder.maxEuclideanDistance", new Option_Float(-1));
73 oc.addDescription("device.stationfinder.maxEuclideanDistance", "Battery", TL("Euclidean search distance in meters (a negative value disables the restriction)"));
74 oc.doRegister("device.stationfinder.repeat", new Option_String("60", "TIME"));
75 oc.addDescription("device.stationfinder.repeat", "Battery", TL("When to trigger a new search if no station has been found"));
76 oc.doRegister("device.stationfinder.maxChargePower", new Option_Float(100000.));
77 oc.addDescription("device.stationfinder.maxChargePower", "Battery", TL("The maximum charging speed of the vehicle battery"));
78 oc.doRegister("device.stationfinder.chargeType", new Option_String("charging"));
79 oc.addDescription("device.stationfinder.chargeType", "Battery", TL("Type of energy transfer"));
80 oc.doRegister("device.stationfinder.waitForCharge", new Option_String("600", "TIME"));
81 oc.addDescription("device.stationfinder.waitForCharge", "Battery", TL("After this waiting time vehicle searches for a new station when the initial one is blocked"));
82 oc.doRegister("device.stationfinder.minOpportunityDuration", new Option_String("3600", "TIME"));
83 oc.addDescription("device.stationfinder.minOpportunityDuration", "Battery", TL("Only stops with a predicted duration of at least the given threshold are considered for opportunistic charging."));
84 oc.doRegister("device.stationfinder.saturatedChargeLevel", new Option_Float(0.8));
85 oc.addDescription("device.stationfinder.saturatedChargeLevel", "Battery", TL("Target state of charge after which the vehicle stops charging"));
86 oc.doRegister("device.stationfinder.needToChargeLevel", new Option_Float(0.4));
87 oc.addDescription("device.stationfinder.needToChargeLevel", "Battery", TL("State of charge the vehicle begins searching for charging stations"));
88 oc.doRegister("device.stationfinder.opportunisticChargeLevel", new Option_Float(0.));
89 oc.addDescription("device.stationfinder.opportunisticChargeLevel", "Battery", TL("State of charge below which the vehicle may look for charging opportunities along its planned stops"));
90 oc.doRegister("device.stationfinder.replacePlannedStop", new Option_Float(0.));
91 oc.addDescription("device.stationfinder.replacePlannedStop", "Battery", TL("Share of stopping time of the next independently planned stop to use for charging instead"));
92 oc.doRegister("device.stationfinder.maxDistanceToReplacedStop", new Option_Float(300.));
93 oc.addDescription("device.stationfinder.maxDistanceToReplacedStop", "Battery", TL("Maximum distance in meters from the original stop to be replaced by the charging stop"));
94 oc.doRegister("device.stationfinder.chargingStrategy", new Option_String("none"));
95 oc.addDescription("device.stationfinder.chargingStrategy", "Battery", TL("Set a charging strategy to alter time and charging load from the set: [none, balanced, latest]"));
96 oc.doRegister("device.stationfinder.checkEnergyForRoute", new Option_Bool(true));
97 oc.addDescription("device.stationfinder.checkEnergyForRoute", "Battery", TL("Only search for charging stations if the battery charge is not estimated sufficient to complete the current route"));
98}
99
100
101
102void
103MSDevice_StationFinder::buildVehicleDevices(SUMOVehicle& v, std::vector<MSVehicleDevice*>& into) {
105 if (equippedByDefaultAssignmentOptions(oc, "stationfinder", v, false)) {
106 into.push_back(new MSDevice_StationFinder(v));
107 }
108}
109
110
111// ---------------------------------------------------------------------------
112// MSDevice_StationFinder-methods
113// ---------------------------------------------------------------------------
115 : MSVehicleDevice(holder, "stationfinder_" + holder.getID()),
116 MSStoppingPlaceRerouter("device.stationfinder.charging", true, {
117 {"waitingTime", 1.}, {"chargingTime", 1.}
118}, { {"waitingTime", false}, {"chargingTime", false} }),
119myVeh(dynamic_cast<MSVehicle&>(holder)),
120myBattery(nullptr), myChargingStation(nullptr), myRescueCommand(nullptr), myChargeLimitCommand(nullptr),
121myLastChargeCheck(0), myCheckInterval(1000), myArrivalAtChargingStation(-1), myLastSearch(-1),
122myLastOpportunisticSearch(-1) {
123 // consider whole path to/from a charging station in the search
124 myEvalParams["distanceto"] = 0.;
125 myEvalParams["timeto"] = 1.;
126 myEvalParams["timefrom"] = 1.;
127 myNormParams["chargingTime"] = true;
128 myNormParams["waitingTime"] = true;
129 myRescueTime = STEPS2TIME(holder.getTimeParam("device.stationfinder.rescueTime"));
130 const std::string chargingStrategy = holder.getStringParam("device.stationfinder.chargingStrategy");
131 if (chargingStrategy == "balanced") {
132 myChargingStrategy = CHARGINGSTRATEGY_BALANCED;
133 } else if (chargingStrategy == "latest") {
134 myChargingStrategy = CHARGINGSTRATEGY_LATEST;
135 } else if (chargingStrategy == "none") {
136 myChargingStrategy = CHARGINGSTRATEGY_NONE;
137 } else {
138 WRITE_ERRORF(TL("Invalid device.stationfinder.chargingStrategy '%'."), chargingStrategy);
139 }
140 const std::string rescueAction = holder.getStringParam("device.stationfinder.rescueAction");
141 if (rescueAction == "remove") {
142 myRescueAction = RESCUEACTION_REMOVE;
143 } else if (rescueAction == "tow") {
144 myRescueAction = RESCUEACTION_TOW;
145 } else if (rescueAction == "none") {
146 myRescueAction = RESCUEACTION_NONE;
147 } else {
148 WRITE_ERRORF(TL("Invalid device.stationfinder.rescueAction '%'."), rescueAction);
149 }
150 initRescueCommand();
151 myReserveFactor = MAX2(1., holder.getFloatParam("device.stationfinder.reserveFactor"));
152 myEmptySoC = MAX2(0., MIN2(holder.getFloatParam("device.stationfinder.emptyThreshold"), 1.));
153 myRadius = holder.getTimeParam("device.stationfinder.radius");
154 myMaxEuclideanDistance = holder.getFloatParam("device.stationfinder.maxEuclideanDistance");
155 myRepeatInterval = holder.getTimeParam("device.stationfinder.repeat");
156 myMaxChargePower = holder.getFloatParam("device.stationfinder.maxChargePower");
157 myChargeType = CHARGETYPE_CHARGING;
158
159 myWaitForCharge = holder.getTimeParam("device.stationfinder.waitForCharge");
160 myMinOpportunisticTime = holder.getTimeParam("device.stationfinder.minOpportunityDuration");
161 myTargetSoC = MAX2(0., MIN2(holder.getFloatParam("device.stationfinder.saturatedChargeLevel"), 1.));
162 mySearchSoC = MAX2(0., MIN2(holder.getFloatParam("device.stationfinder.needToChargeLevel"), 1.));
163 if (mySearchSoC <= myEmptySoC) {
164 WRITE_WARNINGF(TL("Vehicle '%' searches for charging stations only in the rescue case due to search threshold % <= rescue threshold %."), myHolder.getID(), mySearchSoC, myEmptySoC);
165 }
166 myOpportunitySoC = MAX2(0., MIN2(holder.getFloatParam("device.stationfinder.opportunisticChargeLevel"), 1.));
167 if (myOpportunitySoC > 0. && myOpportunitySoC < MIN2(mySearchSoC + NUMERICAL_EPS, 1.)) {
168 myOpportunitySoC = 0.;
169 WRITE_WARNINGF(TL("Vehicle '%' won't do opportunistic charging as the threshold % is too close to the regular one %."), myHolder.getID(), myOpportunitySoC, mySearchSoC);
170 }
171 myReplacePlannedStop = MAX2(0., holder.getFloatParam("device.stationfinder.replacePlannedStop"));
172 myDistanceToOriginalStop = holder.getFloatParam("device.stationfinder.maxDistanceToReplacedStop");
173 myUpdateSoC = 2; // check once at the beginning
174 myCheckEnergyForRoute = holder.getBoolParam("device.stationfinder.checkEnergyForRoute");
175}
176
177
179 // make the rescue command invalid if there is one
180 if (myRescueCommand != nullptr) {
182 }
183 if (myChargeLimitCommand != nullptr) {
185 }
186}
187
188
189bool
190MSDevice_StationFinder::notifyMove(SUMOTrafficObject& veh, double /*oldPos*/, double /*newPos*/, double /*newSpeed*/) {
191 if (myBattery->getEnergyCharged() > 0. && myChargingStation != nullptr) {
193 myChargingStation = nullptr;
195 return true;
196 } else if (mySearchState == SEARCHSTATE_CHARGING) {
197 if (myBattery->getChargingStation() == nullptr) {
199 } else {
200 return true;
201 }
202 }
203 // check if the vehicle travels at most an edge length to the charging station after jump/teleport
205 return true;
206 }
207 const SUMOTime now = SIMSTEP;
208 if (myChargingStation != nullptr) {
210 // waited for too long, try another charging station
211 if (rerouteToChargingStation(true)) {
212 WRITE_MESSAGE(TLF("Rerouted vehicle '%' after waiting too long at the previous charging station at time=%.", veh.getID(), toString(SIMTIME)));
213 }
215 // remember when the vehicle arrived close to the target charging station
218 }
219 }
221 if (currentSoC < myOpportunitySoC && currentSoC < myTargetSoC && mySearchState == SEARCHSTATE_NONE) {
222 // battery SoC is low enough to allow opportunistic charging (charging whenever - wherever)
225 return true;
226 } else if (currentSoC > mySearchSoC || mySearchState == SEARCHSTATE_BROKEN_DOWN) {
227 // battery SoC is too high to look for charging facilities or the vehicle is already in rescue mode
228 return true;
229 }
230 // only check once per second
231 if (now - myLastChargeCheck < 1000) {
232 return true;
233 } else if (myRescueAction != RESCUEACTION_NONE && (currentSoC < myEmptySoC || currentSoC < NUMERICAL_EPS)) {
234
235 // vehicle has to stop at the end of the because battery SoC is too low
236 double brakeGap = myVeh.getCarFollowModel().brakeGap(myVeh.getSpeed());
237 std::pair<const MSLane*, double> stopPos = myVeh.getLanePosAfterDist(brakeGap);
238 if (stopPos.first != nullptr) {
239 const MSLane* stopLane = (stopPos.first->isInternal()) ? stopPos.first->getNormalSuccessorLane() : stopPos.first;
240 double endPos = stopPos.second;
241 if (stopLane != stopPos.first) {
242 endPos = MIN2(POSITION_EPS, stopLane->getLength());
243 }
244 // remove possibly scheduled charging stop
245 if (myVeh.hasStops() && myVeh.getStop(0).chargingStation != nullptr) {
247 }
248
249 // schedule the rescue stop
251 rescueStop.index = 0;
252 rescueStop.edge = stopLane->getEdge().getID();
253 rescueStop.lane = stopLane->getID();
254 rescueStop.startPos = MAX2(endPos - 2 * myHolder.getVehicleType().getLength(), 0.);
255 rescueStop.endPos = endPos;
257 WRITE_MESSAGEF(TL("Vehicle '%' wants to stop on lane % at pos % because of low battery charge % at time=%."), myHolder.getID(), rescueStop.lane, toString(rescueStop.endPos), toString(currentSoC), toString(SIMTIME));
258
260 // remove vehicle from network
261 rescueStop.until = SUMOTime_MAX;
262 rescueStop.breakDown = true;
263 std::string errorMsg = "Could not insert the rescue stop.";
264 if (!myVeh.insertStop(0, rescueStop, "stationfinder:rescue", false, errorMsg)) {
265 WRITE_ERROR(errorMsg);
266 }
268 return true;
269 } else if (myRescueAction == RESCUEACTION_TOW) {
270 // wait next to the road and get teleported to a charging station
271 SUMOTime rescueTime = TIME2STEPS(myRescueTime);
272 rescueStop.duration = rescueTime;
273 rescueStop.parking = ParkingType::ONROAD;
274 rescueStop.jump = 0;
275 std::string errorMsg = "Could not insert the rescue stop.";
276 if (!myVeh.insertStop(0, rescueStop, "stationfinder:rescue", false, errorMsg)) {
277 WRITE_ERROR(errorMsg);
278 }
282 return true;
283 }
284 }
285 } else if (myChargingStation == nullptr &&
288 // check if a charging stop is already planned without the device, otherwise reroute inside this device
289 if (!alreadyPlannedCharging() && now > myHolder.getDeparture()) {
291 }
292 myUpdateSoC = currentSoC - MAX2(0.1 * currentSoC, 0.01);
293#ifdef DEBUG_STATIONFINDER_REROUTE
294 std::cout << SIMTIME << " " << myHolder.getID() << " currentSoC=" << currentSoC << " nextUpdateSoC=" << myUpdateSoC << "\n";
295#endif
296 }
298 return true;
299}
300
301
302bool
306
307
308void
312 std::vector<std::string> internals;
313 internals.push_back(toString(myLastChargeCheck));
314 internals.push_back(toString(myUpdateSoC));
315 internals.push_back(toString(mySearchSoC));
316 internals.push_back(toString(myTargetSoC));
317 internals.push_back(toString(myWaitForCharge));
318 internals.push_back(toString(myRepeatInterval));
319 internals.push_back(toString(myRadius));
320 internals.push_back(toString(myLastSearch));
321 internals.push_back(toString(myReserveFactor));
322 internals.push_back(toString(mySearchState));
323 internals.push_back(toString(myArrivalAtChargingStation));
324 internals.push_back((myChargingStation == nullptr) ? "NULL" : myChargingStation->getID());
325 internals.push_back(toString(myChargeLimits.size()));
326 for (auto chargeLimit : myChargeLimits) {
327 internals.push_back(toString(chargeLimit.first));
328 internals.push_back(toString(chargeLimit.second));
329 }
330 internals.push_back(toString(myOpportunitySoC));
331 internals.push_back(toString(myMinOpportunisticTime));
332 internals.push_back(toString(myCheckEnergyForRoute));
333 out.writeAttr(SUMO_ATTR_STATE, toString(internals));
334 out.closeTag();
335}
336
337
338void
340 std::istringstream bis(attrs.getString(SUMO_ATTR_STATE));
341 bis >> myLastChargeCheck;
342 bis >> myUpdateSoC;
343 bis >> mySearchSoC;
344 bis >> myTargetSoC;
345 bis >> myWaitForCharge;
346 bis >> myRepeatInterval;
347 bis >> myRadius;
348 bis >> myLastSearch;
349 bis >> myReserveFactor;
350 int searchState;
351 bis >> searchState;
352 mySearchState = (SearchState)searchState;
354 std::string csID;
355 bis >> csID;
356 if (csID != "NULL") {
358 }
359 int chargeLimitCount = 0;
360 bis >> chargeLimitCount;
361 for (int i = 0; i < chargeLimitCount; ++i) {
362 SUMOTime t = 0;
363 double limit = 0.;
364 bis >> t;
365 bis >> limit;
366 myChargeLimits.push_back({ t, limit });
367 }
368 bis >> myOpportunitySoC;
371}
372
373
374void
376 const double /* frontOnLane */,
377 const double /* timeOnLane */,
378 const double /* meanSpeedFrontOnLane */,
379 const double /* meanSpeedVehicleOnLane */,
380 const double /* travelledDistanceFrontOnLane */,
381 const double /* travelledDistanceVehicleOnLane */,
382 const double /* meanLengthOnLane */) {
383
384 // called by meso (see MSMeanData_Emissions::MSLaneMeanDataValues::notifyMoveInternal)
385}
386
387
389MSDevice_StationFinder::findChargingStation(SUMOAbstractRouter<MSEdge, SUMOVehicle>& /*router*/, double expectedConsumption, StoppingPlaceParamMap_t& scores, bool constrainTT, bool skipVisited, bool skipOccupied, bool visible) {
390 MSChargingStation* minStation = nullptr;
391 std::vector<StoppingPlaceVisible> candidates;
392 const StoppingPlaceMemory* chargingMemory = myVeh.getChargingMemory();
393 if (chargingMemory == nullptr) {
394 skipVisited = false;
395 }
396 const SUMOTime stoppingPlaceMemory = TIME2STEPS(getWeight(myHolder, "memory", 600));
398 MSChargingStation* cs = static_cast<MSChargingStation*>(stop.second);
399 if (cs->getEfficency() < NUMERICAL_EPS || cs->getChargingPower(false) < NUMERICAL_EPS) {
400 continue;
401 }
402 if (cs->getChargeType() != myBattery->getChargeType()) {
403 continue;
404 }
405 if (cs->getParkingArea() != nullptr && !cs->getParkingArea()->accepts(&myVeh)) {
406 // skip stations where the linked parking area does not grant access to the device holder
407 continue;
408 }
409 if (skipOccupied && freeSpaceAtChargingStation(cs) < 1.) {
410 continue;
411 }
412 if (skipVisited && chargingMemory->sawBlockedStoppingPlace(cs, false) > 0 && SIMSTEP - chargingMemory->sawBlockedStoppingPlace(cs, false) < stoppingPlaceMemory) {
413 // skip recently visited
414 continue;
415 }
416 if (constrainTT && myMaxEuclideanDistance > 0 && stop.second->getLane().geometryPositionAtOffset(stop.second->getBeginLanePosition()).distanceTo2D(myHolder.getPosition()) > myMaxEuclideanDistance) {
417 // skip probably too distant charging stations
418 continue;
419 }
420 if (visible && myHolder.getEdge()->getID() != stop.second->getLane().getEdge().getID()) {
421 continue;
422 }
423 candidates.push_back({cs, false});
424 }
425 ConstMSEdgeVector newRoute;
426 scores["expectedConsumption"] = expectedConsumption;
427 std::vector<double> probs(candidates.size(), 1.);
428 bool newDestination;
429 myCheckValidity = constrainTT;
430 MSStoppingPlace* bestCandidate = rerouteStoppingPlace(nullptr, candidates, probs, myHolder, newDestination, newRoute, scores);
431 myCheckValidity = true;
432 minStation = dynamic_cast<MSChargingStation*>(bestCandidate);
433 return minStation;
434}
435
436
437bool
441#ifdef DEBUG_STATIONFINDER_REROUTE
442 std::cout << SIMTIME << " " << myHolder.getID() << " expectedConsumption=" << expectedConsumption << " reserve=" << (myEmptySoC * myBattery->getMaximumBatteryCapacity()) << " chargeLevel=" << myBattery->getActualBatteryCapacity() << "\n";
443#endif
447 StoppingPlaceParamMap_t scores = {};
448 MSChargingStation* cs = findChargingStation(router, expectedConsumption, scores);
449 if (cs != nullptr) {
450 // integrate previously planned stops which do not have charging facilities
453 stopPar.chargingStation = cs->getID();
454 if (cs->getParkingArea() != nullptr) {
455 stopPar.parkingarea = cs->getParkingArea()->getID();
457 }
458 stopPar.edge = cs->getLane().getEdge().getID();
459 stopPar.lane = cs->getLane().getID();
460 stopPar.duration = TIME2STEPS(expectedConsumption / (cs->getChargingPower(false) * cs->getEfficency()));
462 if (myReplacePlannedStop > 0) {
463 // "reuse" a previously planned stop (stop at charging station instead of a different stop)
464 // what if the charging station is skipped due to long waiting time?
466 // compare the distance to the original target
467 if (scores["distfrom"] < myDistanceToOriginalStop /*actualDist < myDistanceToOriginalStop*/) {
468 // compute the arrival time at the original stop
469 const SUMOTime timeToOriginalStop = TIME2STEPS(scores["timefrom"]);
470 const SUMOTime originalUntil = myHolder.getNextStopParameter()->until;
471 if (timeToOriginalStop + myLastSearch < originalUntil) {
472 const SUMOTime delta = originalUntil - (timeToOriginalStop + myLastSearch);
473 stopPar.until = timeToOriginalStop + myLastSearch + (SUMOTime)((double)delta * MIN2(myReplacePlannedStop, 1.));
474 stopPar.parametersSet |= STOP_UNTIL_SET;
475 if (myReplacePlannedStop > 1.) {
477 }
478 // optionally implement a charging strategy by adjusting the accepted charging rates
480 // the charging strategy should actually only be computed at the arrival at the charging station
481 implementChargingStrategy(myLastSearch + TIME2STEPS(scores["timeto"]), stopPar.until, expectedConsumption, cs);
482 }
483 }
484 }
485 }
486 }
487 stopPar.startPos = cs->getBeginLanePosition();
488 stopPar.endPos = cs->getEndLanePosition();
489 std::string errorMsg;
490#ifdef DEBUG_STATIONFINDER_REROUTE
491 std::ostringstream os;
492 const ConstMSEdgeVector edgesBefore = myVeh.getRoute().getEdges();
493 for (auto edge : edgesBefore) {
494 os << edge->getID() << " ";
495 }
496 std::cout << "MSDevice_StationFinder::rerouteToChargingStation: \n\tRoute before scheduling the charging station: " << os.str() << "\n";
497#endif
498 if ((replace && !myVeh.replaceStop(0, stopPar, "stationfinder:search", false, errorMsg)) || (!replace && !myVeh.insertStop(0, stopPar, "stationfinder:search", false, errorMsg))) {
499 WRITE_MESSAGE(TLF("Problem with inserting the charging station stop for vehicle %.", myHolder.getID()));
500 WRITE_ERROR(errorMsg);
501 }
502
503#ifdef DEBUG_STATIONFINDER_REROUTE
504 std::ostringstream os2;
505 const ConstMSEdgeVector edgesAfter = myVeh.getRoute().getEdges();
506 for (auto edge : edgesAfter) {
507 os2 << edge->getID() << " ";
508 }
509 std::cout << "\tRoute after scheduling the charging station: " << os2.str() << "\n";
510#endif
513#ifdef DEBUG_STATIONFINDER_REROUTE
514 std::cout << "\tVehicle " << myHolder.getID() << " gets rerouted to charging station " << cs->getID() << " on edge " << stopPar.edge << " at time " << SIMTIME << "\n";
515#endif
516 return true;
517 }
519 WRITE_MESSAGEF(TL("Vehicle '%' wants to charge at time=% but does not find any charging station nearby."), myHolder.getID(), toString(SIMTIME));
520 }
521 return false;
522}
523
524
525bool
527 // check next stop
529 if (myHolder.hasStops() && capacityDelta > 0.) {
530 const MSStop& nextStop = myHolder.getNextStop();
531 if (myHolder.isStopped() || nextStop.chargingStation != nullptr || myHolder.getCurrentRouteEdge() != nextStop.edge ||
533 return false;
534 }
535 // find charging station on same edge
537 StoppingPlaceParamMap_t scores = {};
538 MSChargingStation* cs = findChargingStation(router, 0., scores, true, true, true, true);
539 if (cs != nullptr) {
540 // replace next stop by charging stop
543 stopPar.chargingStation = cs->getID();
544 if (cs->getParkingArea() != nullptr) {
545 stopPar.parkingarea = cs->getParkingArea()->getID();
547 }
548 stopPar.edge = cs->getLane().getEdge().getID();
549 stopPar.lane = cs->getLane().getID();
550 stopPar.startPos = cs->getBeginLanePosition();
551 stopPar.endPos = cs->getEndLanePosition();
552 // copy over depart time from previously planned stop
553 SUMOTime oldUntil = nextStop.getUntil();
554 if (oldUntil > 0) {
555 stopPar.until = oldUntil;
556 stopPar.duration = 0;
557 } else { // set a duration needed to charge to the target SoC
558 stopPar.duration = nextStop.duration;
559 }
560 std::string errorMsg;
561 if (!myVeh.replaceStop(0, stopPar, "stationfinder:opportunisticSearch", false, errorMsg)) {
562 WRITE_ERROR(errorMsg);
563 }
564 return true;
565 }
566 }
567 return false;
568}
569
570
573 // find closest charging station
575 double expectedConsumption = MIN2(estimateConsumption(nullptr, true, STEPS2TIME(myVeh.getStops().front().pars.duration)) * myReserveFactor, myBattery->getMaximumBatteryCapacity() * myTargetSoC);
576 StoppingPlaceParamMap_t scores = {};
577 MSChargingStation* cs = findChargingStation(router, expectedConsumption, scores, false, false, true);
578 if (cs == nullptr) {
579 // continue waiting if all charging stations are occupied
580#ifdef DEBUG_STATIONFINDER_RESCUE
581 std::cout << "MSDevice_StationFinder::teleportToChargingStation: No charging station available to teleport the broken-down vehicle " << myHolder.getID() << " to at time " << SIMTIME << ".\n.";
582#endif
583 // remove the vehicle if teleport to a charging station fails
584 if (myHolder.isStopped()) {
585 MSStop& currentStop = myHolder.getNextStopMutable();
586 currentStop.duration += DELTA_T;
587 SUMOVehicleParameter::Stop& stopPar = const_cast<SUMOVehicleParameter::Stop&>(currentStop.pars);
588 stopPar.jump = -1;
589 stopPar.breakDown = true;
591 WRITE_WARNINGF(TL("There is no charging station available to teleport the vehicle '%' to at time=%. Thus the vehicle will be removed."), myHolder.getID(), toString(SIMTIME));
592 }
593#ifdef DEBUG_STATIONFINDER_RESCUE
594 else {
595#ifdef DEBUG_STATIONFINDER_RESCUE
596 std::cout << "MSDevice_StationFinder::teleportToChargingStation: Rescue stop of " << myHolder.getID() << " ended prematurely before regular end at " << SIMTIME << ".\n.";
597#endif
598 }
599#endif
600 return myRepeatInterval;
601 }
602
603 // teleport to the charging station, stop there for charging
606 stopPar.chargingStation = cs->getID();
607 if (cs->getParkingArea() != nullptr) {
608 stopPar.parkingarea = cs->getParkingArea()->getID();
610 }
611 stopPar.edge = cs->getLane().getEdge().getID();
612 stopPar.lane = cs->getLane().getID();
613 stopPar.startPos = cs->getBeginLanePosition();
614 stopPar.endPos = cs->getEndLanePosition();
615 stopPar.duration = TIME2STEPS(expectedConsumption / (cs->getChargingPower(false) * cs->getEfficency()));
616 std::string errorMsg;
617 if (!myVeh.insertStop(1, stopPar, "stationfinder:search", true, errorMsg)) {
618 WRITE_ERROR(errorMsg);
619 }
621 myRescueCommand = nullptr;
622 return 0;
623}
624
625
626double
627MSDevice_StationFinder::estimateConsumption(const MSEdge* target, const bool includeEmptySoC, const double stopDiscount) const {
628 const SUMOTime now = SIMSTEP;
630 const ConstMSEdgeVector& route = myHolder.getRoute().getEdges();
631 ConstMSEdgeVector::const_iterator targetIt = (target == nullptr) ? route.end() : std::find(route.begin(), route.end(), target) + 1;
632 const ConstMSEdgeVector remainingRoute(route.begin() + myHolder.getRoutePosition(), targetIt);
633 const double remainingTime = router.recomputeCosts(remainingRoute, &myHolder, now);
634 if (now > myHolder.getDeparture()) {
635 const double totalConsumption = myBattery->getTotalConsumption();
636 double expectedConsumption = 0.;
637 double passedTime = STEPS2TIME(now - myHolder.getDeparture());
638 if (totalConsumption > 0. && passedTime - stopDiscount > DEFAULT_CONSUMPTION_ESTIMATE_HISTORY) {
639 expectedConsumption = totalConsumption / (passedTime - stopDiscount) * remainingTime;
640 } else {
641 // fallback consumption rate for vehicles starting with low battery
643 WRITE_ERRORF("The stationfinder device needs emission parameters for range estimation but no emission class has been set for the vehicle '%'", myHolder.getID());
644 }
645 const double speed = MIN2(myHolder.getMaxSpeed(), myHolder.getLane()->getSpeedLimit());
648 expectedConsumption = PollutantsInterface::compute(myVeh.getVehicleType().getEmissionClass(), emType,
649 speed * 0.8, 0., 0., params) * (remainingTime - passedTime);
650 }
651 if (includeEmptySoC) {
653 }
655 return expectedConsumption;
656 }
657 return 0.;
658}
659
660
661double
665
666
667bool
669 if (myChargingStation == nullptr) {
670 auto stops = myHolder.getStops();
671 for (auto stop : stops) {
672 if (stop.chargingStation != nullptr) {
673 // compare whether we'll make it there without intermediate charging
674 double expectedConsumption = estimateConsumption(*stop.edge);
675 if (myBattery->getActualBatteryCapacity() < expectedConsumption) {
676 myChargingStation = stop.chargingStation;
677 return true;
678 }
679 }
680 }
681 }
682 return false;
683}
684
685
686void
692
693
694void
700
701
704 if (myChargeLimits.size() > 0 && myChargeLimits.begin()->first < currentTime - DELTA_T) {
705 myChargeLimits.clear();
706 }
707 if (myChargeLimits.size() > 0) {
708 double chargeLimit = myChargeLimits.begin()->second;
709 myBattery->setChargeLimit(chargeLimit);
710 if (chargeLimit < 0) {
711 WRITE_MESSAGEF(TL("The charging rate limit of vehicle '%' is lifted at time=%"), myHolder.getID(), STEPS2TIME(SIMSTEP));
712 } else {
713 WRITE_MESSAGEF(TL("The charging rate of vehicle '%' is limited to % at time=%"), myHolder.getID(), chargeLimit, STEPS2TIME(SIMSTEP));
714 }
715 myChargeLimits.erase(myChargeLimits.begin());
716 }
717 if (myChargeLimits.size() == 0) {
719 myChargeLimitCommand = nullptr;
720 return 0;
721 } else {
722 return myChargeLimits.begin()->first - currentTime;
723 }
724}
725
726
727void
729 myChargeLimits.clear();
731 const double balancedChargeRate = plannedCharge / STEPS2TIME(end - begin) * 3600.;
732 myChargeLimits.push_back({ begin, balancedChargeRate });
733 myChargeLimits.push_back({ end, -1});
734 } else { // CHARGINGSTRATEGY_LATEST
735 SUMOTime expectedDuration = myBattery->estimateChargingDuration(plannedCharge, cs->getChargingPower(false) * cs->getEfficency());
736 if (end - expectedDuration > begin) {
737 myChargeLimits.push_back({ begin, 0 });
738 myChargeLimits.push_back({ end - expectedDuration, -1 });
739 }
740 }
741 if (myChargeLimits.size() > 0) {
744 }
745}
746
747
748void
750 if (tripinfoOut != nullptr && myChargingStation != nullptr) {
751 tripinfoOut->openTag("stationfinder");
752 tripinfoOut->writeAttr("chargingStation", myChargingStation->getID());
753 tripinfoOut->closeTag();
754 }
755}
756
757
758std::string
759MSDevice_StationFinder::getParameter(const std::string& key) const {
760 if (key == "chargingStation") { // eventually abstract with enum
761 return (myChargingStation == nullptr) ? "" : myChargingStation->getID();
762 } else if (key == "batteryNeed") {
764 } else if (key == "needToChargeLevel") {
765 return toString(mySearchSoC);
766 } else if (key == "saturatedChargeLevel") {
767 return toString(myTargetSoC);
768 } else if (key == "waitForCharge") {
770 } else if (key == "repeat") {
772 } else if (key == "radius") {
773 return toString(myRadius);
774 } else if (key == "reserveFactor") {
776 }
777 throw InvalidArgument(TLF("Parameter '%' is not supported for device of type '%'", key, deviceName()));
778}
779
780
781void
782MSDevice_StationFinder::setParameter(const std::string& key, const std::string& value) {
783 double doubleValue;
784 try {
785 doubleValue = StringUtils::toDouble(value);
786 } catch (NumberFormatException&) {
787 throw InvalidArgument(TLF("Setting parameter '%' requires a number for device of type '%'", key, deviceName()));
788 }
789 if (key == "needToChargeLevel") {
790 mySearchSoC = MAX2(0., MIN2(1., doubleValue));
791 } else if (key == "saturatedChargeLevel") {
792 myTargetSoC = MAX2(0., MIN2(1., doubleValue));
793 } else if (key == "waitForCharge") {
794 myWaitForCharge = TIME2STEPS(MAX2(0., doubleValue));
795 } else if (key == "repeat") {
796 myRepeatInterval = TIME2STEPS(MAX2(0., doubleValue));
797 } else if (key == "radius") {
798 myRadius = TIME2STEPS(MAX2(0., doubleValue));
799 } else if (key == "reserveFactor") {
800 myReserveFactor = MAX2(1., doubleValue);
801 } else {
802 throw InvalidArgument(TLF("Setting parameter '%' is not supported for device of type '%'", key, deviceName()));
803 }
804}
805
806
807bool
808MSDevice_StationFinder::evaluateCustomComponents(SUMOVehicle& /* veh */, double /* brakeGap */, bool /* newDestination */,
809 MSStoppingPlace* alternative, double /* occupancy */, double /* prob */,
811 StoppingPlaceParamMap_t& stoppingPlaceValues,
812 ConstMSEdgeVector& /* newRoute */, ConstMSEdgeVector& /* stoppingPlaceApproach */,
813 StoppingPlaceParamMap_t& /* maxValues */, StoppingPlaceParamMap_t& addInput) {
814 // estimated waiting time and charging time
815 MSChargingStation* cs = dynamic_cast<MSChargingStation*>(alternative);
816 double parkingCapacity = (cs->getParkingArea() != nullptr) ? cs->getParkingArea()->getCapacity() : (cs->getEndLanePosition() - cs->getBeginLanePosition()) / myHolder.getVehicleType().getParameter().length;
817 double freeParkingCapacity = freeSpaceAtChargingStation(cs);
818 stoppingPlaceValues["waitingTime"] = (freeParkingCapacity < 1.) ? DEFAULT_AVG_WAITING_TIME / parkingCapacity : 0.;
819 stoppingPlaceValues["chargingTime"] = STEPS2TIME(cs->getChargeDelay()) + addInput["expectedConsumption"] / cs->getChargingPower(false);
820 return true;
821}
822
823
824bool
826 if (stoppingPlaceValues["timeto"] > STEPS2TIME(myRadius)) {
827 return false;
828 }
829 return true;
830}
831
832
833bool
835 return true;
836}
837
838
842
843
844double
846 MSChargingStation* cs = dynamic_cast<MSChargingStation*>(stoppingPlace);
847 if (cs->getParkingArea() != nullptr) {
848 return cs->getParkingArea()->getOccupancy();
849 }
851}
852
853
854double
856 MSChargingStation* cs = dynamic_cast<MSChargingStation*>(stoppingPlace);
857 if (cs->getParkingArea() != nullptr) {
858 return cs->getParkingArea()->getLastStepOccupancy();
859 }
861}
862
863
864double
866 MSChargingStation* cs = dynamic_cast<MSChargingStation*>(stoppingPlace);
867 if (cs->getParkingArea() != nullptr) {
868 return cs->getParkingArea()->getCapacity();
869 }
871}
872
873
874void
876 veh.rememberBlockedChargingStation(stoppingPlace, blocked);
877}
878
879
880void
882 veh.rememberChargingStationScore(place, score);
883}
884
885
886void
890
891
896
897
898int
902
903
904void
907
908
909/****************************************************************************/
long long int SUMOTime
Definition GUI.h:36
#define DEFAULT_CONSUMPTION_ESTIMATE_HISTORY
#define DEFAULT_AVG_WAITING_TIME
#define DEFAULT_CHARGINGSTATION_VIEW_DIST
std::vector< const MSEdge * > ConstMSEdgeVector
Definition MSEdge.h:74
#define WRITE_WARNINGF(...)
Definition MsgHandler.h:288
#define WRITE_MESSAGEF(...)
Definition MsgHandler.h:290
#define WRITE_ERRORF(...)
Definition MsgHandler.h:297
#define WRITE_MESSAGE(msg)
Definition MsgHandler.h:289
#define WRITE_ERROR(msg)
Definition MsgHandler.h:296
#define TL(string)
Definition MsgHandler.h:305
#define TLF(string,...)
Definition MsgHandler.h:307
SUMOTime DELTA_T
Definition SUMOTime.cpp:38
#define STEPS2TIME(x)
Definition SUMOTime.h:55
#define SIMSTEP
Definition SUMOTime.h:61
#define SUMOTime_MAX
Definition SUMOTime.h:34
#define SIMTIME
Definition SUMOTime.h:62
#define TIME2STEPS(x)
Definition SUMOTime.h:57
const long long int VTYPEPARS_EMISSIONCLASS_SET
const int STOP_DURATION_SET
const int STOP_UNTIL_SET
const int STOP_START_SET
const int STOP_END_SET
@ SUMO_TAG_DEVICE
@ SUMO_TAG_CHARGING_STATION
A Charging Station.
@ SUMO_ATTR_ID
@ SUMO_ATTR_PROPULSIONEFFICIENCY
Propulsion efficiency.
@ SUMO_ATTR_STATE
The state of a link.
T MIN2(T a, T b)
Definition StdDefs.h:80
T MAX2(T a, T b)
Definition StdDefs.h:86
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:46
An upper class for objects with additional parameters.
double getDoubleOptional(SumoXMLAttr attr, const double def) const
Returns the value for a given key with an optional default. SUMO_ATTR_MASS and SUMO_ATTR_FRONTSURFACE...
bool replaceStop(int nextStopIndex, SUMOVehicleParameter::Stop stop, const std::string &info, bool teleport, std::string &errorMsg)
const std::list< MSStop > & getStops() const
bool hasStops() const
Returns whether the vehicle has to stop somewhere.
MSStop & getStop(int nextStopIndex)
bool insertStop(int nextStopIndex, SUMOVehicleParameter::Stop stop, const std::string &info, bool teleport, std::string &errorMsg)
const MSRoute & getRoute() const
Returns the current route.
const StoppingPlaceMemory * getChargingMemory() const
const MSVehicleType & getVehicleType() const
Returns the vehicle's type definition.
bool abortNextStop(int nextStopIndex=0)
deletes the next stop at the given index if it exists
double brakeGap(const double speed) const
Returns the distance the vehicle needs to halt including driver's reaction time tau (i....
Definition MSCFModel.h:408
double getChargingPower(bool usingFuel) const
Get charging station's charging power.
SUMOTime getChargeDelay() const
Get Charge Delay.
double getEfficency() const
Get efficiency of the charging station.
ChargeType getChargeType() const
Get charge type.
const MSParkingArea * getParkingArea() const
Get the parking area the charging station is placed on.
MSChargingStation::ChargeType getChargeType() const
Get the charge type.
double getActualBatteryCapacity() const
Get the actual vehicle's Battery Capacity in Wh.
void setChargeLimit(const double limit)
Set (temporary) charge limit.
MSChargingStation * getChargingStation() const
double getMaximumBatteryCapacity() const
Get the total vehicle's Battery Capacity in Wh.
SUMOTime estimateChargingDuration(const double toCharge, const double csPower) const
Estimate the charging duration given the current battery state.
bool tracksFuel() const
Whether the battery device is actually used as a tank of a combustion vehicle.
double getTotalConsumption() const
Get total consumption.
double getEnergyCharged() const
Get charged energy.
A device which triggers rerouting to nearby charging stations.
double myUpdateSoC
SoC the last time the station finder algorithm was run completely.
void resetStoppingPlaceScores(SUMOVehicle &veh) override
forget all stopping place score for this vehicle
void setParameter(const std::string &key, const std::string &value) override
try to set the given parameter for this device. Throw exception for unsupported key
void notifyMoveInternal(const SUMOTrafficObject &veh, const double frontOnLane, const double timeOnLane, const double meanSpeedFrontOnLane, const double meanSpeedVehicleOnLane, const double travelledDistanceFrontOnLane, const double travelledDistanceVehicleOnLane, const double meanLengthOnLane) override
Internal notification about the vehicle moves, see MSMoveReminder::notifyMoveInternal()
void rememberBlockedStoppingPlace(SUMOVehicle &veh, const MSStoppingPlace *stoppingPlace, bool blocked) override
store the blocked stopping place in the vehicle
WrappingCommand< MSDevice_StationFinder > * myRescueCommand
The command responsible for rescue actions.
void implementChargingStrategy(SUMOTime begin, SUMOTime end, const double plannedCharge, const MSChargingStation *cs)
SUMOTime myLastOpportunisticSearch
Last time charging stations have been searched for opportunistic charging.
MSDevice_StationFinder(SUMOVehicle &holder)
Constructor.
ChargingStrategy myChargingStrategy
The chosen charging strategy.
MSChargingStation * findChargingStation(SUMOAbstractRouter< MSEdge, SUMOVehicle > &router, double expectedConsumption, StoppingPlaceParamMap_t &scores, bool constrainTT=true, bool skipVisited=true, bool skipOccupied=false, bool visible=false)
central search function for close charging stations
SUMOTime teleportToChargingStation(const SUMOTime currentTime)
search for a charging station and teleport the vehicle there as a rescue measure
double myDistanceToOriginalStop
The distance in meters to the original stop replaced by the charging stop (models charging close to t...
MSVehicle & myVeh
myHolder cast to needed type
double getStoppingPlaceOccupancy(MSStoppingPlace *stoppingPlace) override
Return the number of occupied places of the StoppingPlace.
MSStoppingPlace * myChargingStation
To which station we are currently travelling.
WrappingCommand< MSDevice_StationFinder > * myChargeLimitCommand
The command responsible for limiting the charging rate (~ implement charging strategies)
static void insertOptions(OptionsCont &oc)
Inserts MSDevice_StationFinder-options.
int getNumberStoppingPlaceReroutes(SUMOVehicle &veh) override
ask how many times already the vehicle has been rerouted to another stopping place
double myTargetSoC
The target state of charge where the vehicle stops charging.
const std::string deviceName() const override
return the name for this type of device
std::string getParameter(const std::string &key) const override
try to retrieve the given parameter from this device. Throw exception for unsupported key
double myOpportunitySoC
The state of charge at/below which the vehicle is interested in charging although it may still be suf...
bool rerouteToChargingStation(bool replace=false)
reroute to a charging station
double myRescueTime
The time to wait for a rescue vehicle in case the battery is empty.
SUMOTime myLastChargeCheck
Last time the SoC was checked.
static void buildVehicleDevices(SUMOVehicle &v, std::vector< MSVehicleDevice * > &into)
Build devices for the given vehicle, if needed.
void initRescueCommand()
create the event command for teleporting in case of brake-down
bool useStoppingPlace(MSStoppingPlace *stoppingPlace) override
Whether the stopping place should be included in the search (can be used to add an additional filter)
double getLastStepStoppingPlaceOccupancy(MSStoppingPlace *stoppingPlace) override
Return the number of occupied places of the StoppingPlace from the previous time step.
double getStoppingPlaceCapacity(MSStoppingPlace *stoppingPlace) override
Return the number of places the StoppingPlace provides.
void rememberStoppingPlaceScore(SUMOVehicle &veh, MSStoppingPlace *place, const std::string &score) override
store the stopping place score in the vehicle
void saveState(OutputDevice &out) const override
Saves the state of the device.
bool myCheckEnergyForRoute
Whether to skip searching charging stations if the battery charge is sufficient to complete the curre...
SUMOTime myWaitForCharge
Accepted waiting time at the charging station before a place becomes available.
SUMOTime myLastSearch
Last time charging stations have been searched.
void setNumberStoppingPlaceReroutes(SUMOVehicle &veh, int value) override
update the number of reroutes for the vehicle
SUMOTime sawBlockedStoppingPlace(SUMOVehicle &veh, MSStoppingPlace *place, bool local) override
ask the vehicle when it has seen the stopping place
double mySearchSoC
The state of charge at which the vehicle starts looking for charging stations.
SUMOTime myArrivalAtChargingStation
Arrival time in the vicinity of the target charging station (to track the waiting time before accessi...
double myEmptySoC
The state of charge threshold below which rescue mode is activated.
double myMaxEuclideanDistance
The maximum euclidean distance between the vehicle and the charging station (-1 deactivates the condi...
bool validComponentValues(StoppingPlaceParamMap_t &stoppingPlaceValues) override
Whether the stopping place should be discarded due to its results from the component evaluation.
bool notifyIdle(SUMOTrafficObject &veh) override
Computes idling emission values and adds them to the emission sums.
void generateOutput(OutputDevice *tripinfoOut) const override
Called on writing tripinfo output.
MSDevice_Battery * myBattery
The corresponding battery device.
bool alreadyPlannedCharging()
adopt a planned charging stop outside of the device
double freeSpaceAtChargingStation(MSChargingStation *cs) const
compute the free space at a charging station
std::vector< std::pair< SUMOTime, double > > myChargeLimits
The next charging rates to set via myChargingRateCommand.
SUMOTime myRadius
The max travel time to the next charging station.
SearchState mySearchState
The current state of the charging search (remember for decision logic)
SUMOTime myRepeatInterval
Time interval to search again for a charging station if the first attempt failed.
bool evaluateCustomComponents(SUMOVehicle &veh, double brakeGap, bool newDestination, MSStoppingPlace *alternative, double occupancy, double prob, SUMOAbstractRouter< MSEdge, SUMOVehicle > &router, StoppingPlaceParamMap_t &stoppingPlaceValues, ConstMSEdgeVector &newRoute, ConstMSEdgeVector &stoppingPlaceApproach, StoppingPlaceParamMap_t &maxValues, StoppingPlaceParamMap_t &addInput) override
Compute some custom target function components.
bool planOpportunisticCharging()
check which stop is suited for opportunistic charging and try to plan charging stops
SUMOTime updateChargeLimit(const SUMOTime currentTime)
update the maximum charge rate of the battery to simulate charging strategies
SUMOTime myMinOpportunisticTime
Minimal expected stop duration to allow for opportunistic charging (not needed to complete the route)
double myReplacePlannedStop
The share of stopping time a charging stop should take from the next regular (non-charging) stop unde...
RescueAction myRescueAction
What to do when the state of charge gets very low.
bool notifyMove(SUMOTrafficObject &veh, double oldPos, double newPos, double newSpeed) override
Computes current emission values and adds them to their sums.
void loadState(const SUMOSAXAttributes &attrs) override
Loads the state of the device from the given description.
double myReserveFactor
The safety buffer when calculating expected consumption.
void initChargeLimitCommand()
create the event command for changing charging rates
double estimateConsumption(const MSEdge *target=nullptr, const bool includeEmptySoC=true, const double stopDiscount=0.) const
estimate the energy needed for the planned route / up to a target edge
SUMOAbstractRouter< MSEdge, SUMOVehicle > & getRouter(SUMOVehicle &veh, const Prohibitions &prohibited) override
Provide the router to use (MSNet::getRouterTT or MSRoutingEngine)
static void insertDefaultAssignmentOptions(const std::string &deviceName, const std::string &optionsTopic, OptionsCont &oc, const bool isPerson=false)
Adds common command options that allow to assign devices to vehicles.
Definition MSDevice.cpp:157
static bool equippedByDefaultAssignmentOptions(const OptionsCont &oc, const std::string &deviceName, DEVICEHOLDER &v, bool outputOptionSet, const bool isPerson=false)
Determines whether a vehicle should get a certain device.
Definition MSDevice.h:200
A road/street connecting two junctions.
Definition MSEdge.h:77
virtual void addEvent(Command *operation, SUMOTime execTimeStep=-1)
Adds an Event.
Representation of a lane in the micro simulation.
Definition MSLane.h:84
const MSLane * getNormalSuccessorLane() const
get normal lane following this internal lane, for normal lanes, the lane itself is returned
Definition MSLane.cpp:3240
double getSpeedLimit() const
Returns the lane's maximum allowed speed.
Definition MSLane.h:597
double getLength() const
Returns the lane's length.
Definition MSLane.h:611
MSEdge & getEdge() const
Returns the lane's edge.
Definition MSLane.h:769
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:477
MSStoppingPlace * getStoppingPlace(const std::string &id, const SumoXMLTag category) const
Returns the named stopping place of the given category.
Definition MSNet.cpp:1403
const NamedObjectCont< MSStoppingPlace * > & getStoppingPlaces(SumoXMLTag category) const
Definition MSNet.cpp:1438
int getCapacity() const
Returns the area capacity.
bool parkOnRoad() const
whether vehicles park on the road
bool accepts(SUMOVehicle *veh) const
Return the parking accepts the vehicle (due to its given badges)
int getLastStepOccupancy() const
Returns the area occupancy at the end of the last simulation step.
int getOccupancy() const
Returns the area occupancy.
const ConstMSEdgeVector & getEdges() const
Definition MSRoute.h:125
static MSVehicleRouter & getRouterTT(const int rngIndex, SUMOVehicleClass svc, const Prohibitions &prohibited={})
return the vehicle router instance
SUMOTime getMinDuration(SUMOTime time) const
return minimum stop duration when starting stop at time
Definition MSStop.cpp:134
MSRouteIterator edge
The edge in the route to stop at.
Definition MSStop.h:48
MSStoppingPlace * chargingStation
(Optional) charging station if one is assigned to the stop
Definition MSStop.h:60
SUMOTime duration
The stopping duration.
Definition MSStop.h:67
SUMOTime getUntil() const
return until / ended time
Definition MSStop.cpp:151
const SUMOVehicleParameter::Stop pars
The stop parameter.
Definition MSStop.h:65
A lane area vehicles can halt at.
double getBeginLanePosition() const
Returns the begin position of this stop.
double getEndLanePosition() const
Returns the end position of this stop.
const MSLane & getLane() const
Returns the lane this stop is located at.
virtual double getLastFreePos(const SUMOVehicle &forVehicle, double brakePos=0) const
Returns the last free position on this stop.
std::map< std::string, double > StoppingPlaceParamMap_t
MSStoppingPlace * rerouteStoppingPlace(MSStoppingPlace *destStoppingPlace, const std::vector< StoppingPlaceVisible > &stoppingPlaceCandidates, const std::vector< double > &probs, SUMOVehicle &veh, bool &newDestination, ConstMSEdgeVector &newRoute, StoppingPlaceParamMap_t &scores, const Prohibitions &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...
double getWeight(SUMOVehicle &veh, const std::string param, const double defaultWeight, const bool warn=false)
read the value of a stopping place search param, e.g. a component weight factor
std::map< const MSEdge *, double > Prohibitions
Abstract in-vehicle device.
SUMOVehicle & myHolder
The vehicle that stores the device.
Representation of a vehicle in the micro simulation.
Definition MSVehicle.h:77
bool willStop() const
Returns whether the vehicle will stop on the current edge.
double getDistanceToPosition(double destPos, const MSLane *destLane) const
const MSLane * getLane() const
Returns the lane the vehicle is on.
Definition MSVehicle.h:581
std::pair< const MSLane *, double > getLanePosAfterDist(double distance) const
return lane and position along bestlanes at the given distance
double getSpeed() const
Returns the vehicle's current speed.
Definition MSVehicle.h:490
const MSCFModel & getCarFollowModel() const
Returns the vehicle's car following model definition.
Definition MSVehicle.h:969
double getMinGap() const
Get the free space in front of vehicles of this class.
SUMOEmissionClass getEmissionClass() const
Get this vehicle type's emission class.
double getLength() const
Get vehicle's length [m].
const SUMOVTypeParameter & getParameter() const
const std::string & getID() const
Returns the id.
Definition Named.h:74
A storage for options typed value containers)
Definition OptionsCont.h:89
void addDescription(const std::string &name, const std::string &subtopic, const std::string &description)
Adds a description for an option.
void doRegister(const std::string &name, Option *o)
Adds an option under the given name.
static OptionsCont & getOptions()
Retrieves the options.
Static storage of an output device and its base (abstract) implementation.
OutputDevice & writeAttr(const SumoXMLAttr attr, const T &val)
writes a named attribute
OutputDevice & openTag(const std::string &xmlElement)
Opens an XML tag.
bool closeTag(const std::string &comment="")
Closes the most recently opened tag and optionally adds a comment.
EmissionType
Enumerating all emission types, including fuel.
static double compute(const SUMOEmissionClass c, const EmissionType e, const double v, const double a, const double slope, const EnergyParams *param)
Returns the amount of the emitted pollutant given the vehicle type and state (in mg/s or ml/s for fue...
virtual double recomputeCosts(const std::vector< const E * > &edges, const V *const v, SUMOTime msTime, double *lengthp=nullptr) const
Encapsulated SAX-Attributes.
virtual std::string getString(int id, bool *isPresent=nullptr) const =0
Returns the string-value of the named (by its enum-value) attribute.
Representation of a vehicle, person, or container.
virtual const MSVehicleType & getVehicleType() const =0
Returns the object's "vehicle" type.
virtual const MSLane * getLane() const =0
Returns the lane the object is currently at.
virtual int getRNGIndex() const =0
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 Position getPosition(const double offset=0) const =0
Return current position (x/y, cartesian)
virtual int getRoutePosition() const =0
return index of edge within route
virtual const MSEdge * getEdge() const =0
Returns the edge the object is currently at.
bool wasSet(long long int what) const
Returns whether the given parameter was set.
double length
The physical vehicle length.
Representation of a vehicle.
Definition SUMOVehicle.h:62
virtual bool isStopped() const =0
Returns whether the vehicle is at a stop and waiting for a person or container to continue.
virtual bool hasStops() const =0
Returns whether the vehicle has to stop somewhere.
virtual const std::list< MSStop > & getStops() const =0
virtual SUMOTime sawBlockedChargingStation(const MSStoppingPlace *cs, bool local) const =0
virtual SUMOTime getDeparture() const =0
Returns this vehicle's real departure time.
virtual double getLength() const =0
Returns the vehicles's length.
virtual void rememberBlockedChargingStation(const MSStoppingPlace *cs, bool local)=0
virtual const MSStop & getNextStop() const =0
virtual MSStop & getNextStopMutable()=0
virtual void resetChargingStationScores()=0
virtual EnergyParams * getEmissionParameters() const =0
Returns the vehicle's emission model parameter.
virtual void rememberChargingStationScore(const MSStoppingPlace *cs, const std::string &score)=0
virtual const ConstMSEdgeVector::const_iterator & getCurrentRouteEdge() const =0
Returns an iterator pointing to the current edge in this vehicles route.
virtual const SUMOVehicleParameter::Stop * getNextStopParameter() const =0
Returns parameters of the next stop or nullptr.
virtual const MSRoute & getRoute() const =0
Returns the current route.
virtual bool abortNextStop(int nextStopIndex=0)=0
deletes the next stop at the given index if it exists
Definition of vehicle stop (position and duration)
std::string edge
The edge to stop at.
ParkingType parking
whether the vehicle is removed from the net while stopping
std::string lane
The lane to stop at.
std::string parkingarea
(Optional) parking area if one is assigned to the stop
double startPos
The stopping position start.
bool breakDown
Whether this stop was triggered by a car failure / mechanical problem / lack of energy.
std::string chargingStation
(Optional) charging station if one is assigned to the stop
int parametersSet
Information for the output which parameter were set.
int index
at which position in the stops list
SUMOTime jump
transfer time if there shall be a jump from this stop to the next route edge
SUMOTime until
The time at which the vehicle may continue its journey.
double endPos
The stopping position end.
SUMOTime duration
The stopping duration.
SUMOTime sawBlockedStoppingPlace(const MSStoppingPlace *stoppingPlace, bool local) const
Get the time the StoppingPlace was confirmed to be blocked.
static double toDouble(const std::string &sData)
converts a string into the double value described by it by calling the char-type converter
A wrapper for a Command function.
void deschedule()
Marks this Command as being descheduled.