Eclipse SUMO - Simulation of Urban MObility
Loading...
Searching...
No Matches
MSAbstractLaneChangeModel.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/****************************************************************************/
23// Interface for lane-change models
24/****************************************************************************/
25
26// ===========================================================================
27// DEBUG
28// ===========================================================================
29//#define DEBUG_TARGET_LANE
30//#define DEBUG_SHADOWLANE
31//#define DEBUG_OPPOSITE
32//#define DEBUG_MANEUVER
33#define DEBUG_COND (myVehicle.isSelected())
34
35#include <config.h>
36
40#include <microsim/MSNet.h>
41#include <microsim/MSEdge.h>
42#include <microsim/MSLane.h>
43#include <microsim/MSLink.h>
44#include <microsim/MSStop.h>
47#include <microsim/MSGlobals.h>
49#include "MSLCM_DK2008.h"
50#include "MSLCM_LC2013.h"
51#include "MSLCM_LC2013_CC.h"
52#include "MSLCM_SL2015.h"
54
55/* -------------------------------------------------------------------------
56 * static members
57 * ----------------------------------------------------------------------- */
63const double MSAbstractLaneChangeModel::NO_NEIGHBOR(std::numeric_limits<double>::max());
65
66#define LC_ASSUMED_DECEL 1.0 // the minimal constant deceleration assumed to estimate the duration of a continuous lane-change at its initiation.
67
68/* -------------------------------------------------------------------------
69 * MSAbstractLaneChangeModel-methods
70 * ----------------------------------------------------------------------- */
71
72void
74 myAllowOvertakingRight = oc.getBool("lanechange.overtake-right");
75 myLCOutput = oc.isSet("lanechange-output");
76 myLCStartedOutput = oc.getBool("lanechange-output.started");
77 myLCEndedOutput = oc.getBool("lanechange-output.ended");
78 myLCXYOutput = oc.getBool("lanechange-output.xy");
79}
80
81
85 throw ProcessError(TLF("Lane change model '%' is not compatible with sublane simulation", toString(lcm)));
86 }
87 switch (lcm) {
89 return new MSLCM_DK2008(v);
91 return new MSLCM_LC2013(v);
93 return new MSLCM_LC2013_CC(v);
95 return new MSLCM_SL2015(v);
98 return new MSLCM_LC2013(v);
99 } else {
100 return new MSLCM_SL2015(v);
101 }
102 default:
103 throw ProcessError(TLF("Lane change model '%' not implemented", toString(lcm)));
104 }
105}
106
107
109 myVehicle(v),
110 myOwnState(0),
111 myPreviousState(0),
112 myPreviousState2(0),
113 myCanceledStateRight(LCA_NONE),
114 myCanceledStateCenter(LCA_NONE),
115 myCanceledStateLeft(LCA_NONE),
116 mySpeedLat(0),
117 myAccelerationLat(0),
118 myAngleOffset(0),
119 myPreviousAngleOffset(0),
120 myCommittedSpeed(0),
121 myLaneChangeCompletion(1.0),
122 myLaneChangeDirection(0),
123 myAlreadyChanged(false),
124 myShadowLane(nullptr),
125 myTargetLane(nullptr),
126 myModel(model),
127 myLastLateralGapLeft(0.),
128 myLastLateralGapRight(0.),
129 myLastLeaderGap(0.),
130 myLastFollowerGap(0.),
131 myLastLeaderSecureGap(0.),
132 myLastFollowerSecureGap(0.),
133 myLastOrigLeaderGap(0.),
134 myLastOrigLeaderSecureGap(0.),
135 myLastLeaderSpeed(0),
136 myLastFollowerSpeed(0),
137 myLastOrigLeaderSpeed(0),
138 myDontResetLCGaps(false),
139 myStrategicLookahead(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_STRATEGIC_LOOKAHEAD, UNDEFINED_LOOKAHEAD)),
140 myMaxSpeedLatStanding(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_MAXSPEEDLATSTANDING, v.getVehicleType().getMaxSpeedLat())),
141 myMaxSpeedLatFactor(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_MAXSPEEDLATFACTOR, 1)),
142 myMaxDistLatStanding(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_MAXDISTLATSTANDING,
143 // prevent lateral sliding for cars but permit for two-wheelers due to better maneuverability
144 (v.getVClass() & (SVC_BICYCLE | SVC_MOTORCYCLE | SVC_MOPED)) != 0 ? std::numeric_limits<double>::max() : 1.6)),
145 mySigma(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_SIGMA, 0.0)),
146 myOvertakeRightParam(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_OVERTAKE_RIGHT, 0)),
147 myAssertive(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_ASSERTIVE, 1)),
148 myCooperativeHelpTime(TIME2STEPS(v.getVehicleType().getParameter().getLCParam(SUMO_ATTR_LCA_COOPERATIVE_HELPTIME, 60))),
149 myHaveBlueLight(v.getDevice(typeid(MSDevice_Bluelight)) != nullptr), // see MSVehicle::initDevices
150 myLastLaneChangeOffset(0),
151 myAmOpposite(false),
152 myManeuverDist(0.),
153 myPreviousManeuverDist(0.) {
157}
158
159
162
163void
166 myOwnState = state;
167 myPreviousState = state; // myOwnState is modified in prepareStep so we make a backup
168}
169
170void
171MSAbstractLaneChangeModel::updateSafeLatDist(const double travelledLatDist) {
172 UNUSED_PARAMETER(travelledLatDist);
173}
174
175
176void
178#ifdef DEBUG_MANEUVER
179 if (DEBUG_COND) {
180 std::cout << SIMTIME
181 << " veh=" << myVehicle.getID()
182 << " setManeuverDist() old=" << myManeuverDist << " new=" << dist
183 << std::endl;
184 }
185#endif
186 myManeuverDist = fabs(dist) < NUMERICAL_EPS ? 0. : dist;
187 // store value which may be modified by the model during the next step
189}
190
191
192double
196
197double
201
202void
204 if (dir == -1) {
205 myLeftFollowers = std::make_shared<MSLeaderDistanceInfo>(followers);
206 myLeftLeaders = std::make_shared<MSLeaderDistanceInfo>(leaders);
207 } else if (dir == 1) {
208 myRightFollowers = std::make_shared<MSLeaderDistanceInfo>(followers);
209 myRightLeaders = std::make_shared<MSLeaderDistanceInfo>(leaders);
210 } else {
211 // dir \in {-1,1} !
212 assert(false);
213 }
214}
215
216
217void
218MSAbstractLaneChangeModel::saveNeighbors(const int dir, const std::pair<MSVehicle* const, double>& follower, const std::pair<MSVehicle* const, double>& leader) {
219 if (dir == -1) {
220 myLeftFollowers = std::make_shared<MSLeaderDistanceInfo>(follower, myVehicle.getLane()->getWidth());
221 myLeftLeaders = std::make_shared<MSLeaderDistanceInfo>(leader, myVehicle.getLane()->getWidth());
222 } else if (dir == 1) {
223 myRightFollowers = std::make_shared<MSLeaderDistanceInfo>(follower, myVehicle.getLane()->getWidth());
224 myRightLeaders = std::make_shared<MSLeaderDistanceInfo>(leader, myVehicle.getLane()->getWidth());
225 } else {
226 // dir \in {-1,1} !
227 assert(false);
228 }
229}
230
231
232void
234 myLeftFollowers = nullptr;
235 myLeftLeaders = nullptr;
236 myRightFollowers = nullptr;
237 myRightLeaders = nullptr;
238}
239
240
241const std::shared_ptr<MSLeaderDistanceInfo>
243 if (dir == -1) {
244 return myLeftFollowers;
245 } else if (dir == 1) {
246 return myRightFollowers;
247 } else {
248 // dir \in {-1,1} !
249 assert(false);
250 }
251 return nullptr;
252}
253
254const std::shared_ptr<MSLeaderDistanceInfo>
256 if (dir == -1) {
257 return myLeftLeaders;
258 } else if (dir == 1) {
259 return myRightLeaders;
260 } else {
261 // dir \in {-1,1} !
262 assert(false);
263 }
264 return nullptr;
265}
266
267
268bool
270 if (neighLeader == nullptr) {
271 return false;
272 }
273 // Congested situation are relevant only on highways (maxSpeed > 70km/h)
274 // and congested on German Highways means that the vehicles have speeds
275 // below 60km/h. Overtaking on the right is allowed then.
276 if ((myVehicle.getLane()->getSpeedLimit() <= 70.0 / 3.6) || (neighLeader->getLane()->getSpeedLimit() <= 70.0 / 3.6)) {
277
278 return false;
279 }
280 if (myVehicle.congested() && neighLeader->congested()) {
281 return true;
282 }
283 return false;
284}
285
286
287bool
288MSAbstractLaneChangeModel::avoidOvertakeRight(const MSVehicle* const neighLeader, const bool allowProb) const {
289 return (!myAllowOvertakingRight // the highway case
290 && !myVehicle.congested()
293 (neighLeader != nullptr && neighLeader->isStopped() // the bus stop case
294 && neighLeader->getStops().front().busstop != nullptr
295 && !StringUtils::toBool(neighLeader->getStops().front().busstop->getParameter("allowOvertakeRight", "true")));
296}
297
298bool
299MSAbstractLaneChangeModel::predInteraction(const std::pair<MSVehicle*, double>& leader) {
300 if (leader.first == 0) {
301 return false;
302 }
303 // let's check it on highways only
304 if (leader.first->getSpeed() < (80.0 / 3.6)) {
305 return false;
306 }
307 return leader.second < getCarFollowModel().interactionGap(&myVehicle, leader.first->getSpeed());
308}
309
310
311bool
315 myLaneChangeDirection = direction;
316 setManeuverDist((target->getWidth() + source->getWidth()) * 0.5 * direction);
319 if (myLCOutput) {
321 }
322 return true;
323 } else {
324 primaryLaneChanged(source, target, direction);
325 return false;
326 }
327}
328
329void
333
334void
338
339void
341 initLastLaneChangeOffset(direction);
343 source->leftByLaneChange(&myVehicle);
344 laneChangeOutput("change", source, target, direction); // record position on the source edge in case of opposite change
345 if (&source->getEdge() != &target->getEdge()) {
347#ifdef DEBUG_OPPOSITE
348 if (debugVehicle()) {
349 std::cout << SIMTIME << " veh=" << myVehicle.getID() << " primaryLaneChanged source=" << source->getID() << " target=" << target->getID() << " nowOpposite=" << myAmOpposite << "\n";
350 }
351#endif
354 } else if (myAmOpposite) {
355#ifdef DEBUG_OPPOSITE
356 if (debugVehicle()) {
357 std::cout << SIMTIME << " veh=" << myVehicle.getID() << " primaryLaneChanged source=" << source->getID() << " target=" << target->getID() << " stayOpposite\n";
358 }
359#endif
360 myAlreadyChanged = true;
362 if (!MSGlobals::gSublane) {
363 // in the continous case, the vehicle is added to the target lane via MSLaneChanger::continueChange / registerHop
364 // in the sublane case, the vehicle is added to the target lane via MSLaneChangerSublane::checkChangeOppositeSublane / MSLane::myTmpVehicles
366 }
367 } else {
370 }
371 // Assure that the drive items are up to date (even if the following step is no actionstep for the vehicle).
372 // This is necessary because the lane advance uses the target lane from the corresponding drive item.
374 changed();
375}
376
377void
378MSAbstractLaneChangeModel::laneChangeOutput(const std::string& tag, MSLane* source, MSLane* target, int direction, double maneuverDist) {
379 if (myLCOutput) {
380 OutputDevice& of = OutputDevice::getDeviceByOption("lanechange-output");
381 of.openTag(tag);
384 of.writeAttr(SUMO_ATTR_TIME, time2string(MSNet::getInstance()->getCurrentTimeStep()));
385 of.writeAttr(SUMO_ATTR_FROM, source->getID());
386 of.writeAttr(SUMO_ATTR_TO, target->getID());
387 of.writeAttr(SUMO_ATTR_DIR, direction);
405 const double latGap = direction < 0 ? myLastLateralGapRight : myLastLateralGapLeft;
406 of.writeAttr("latGap", latGap == NO_NEIGHBOR ? "None" : toString(latGap), latGap == NO_NEIGHBOR);
407 if (maneuverDist != 0) {
408 of.writeAttr("maneuverDistance", toString(maneuverDist));
409 }
410 }
411 if (myLCXYOutput) {
414 }
415 of.closeTag();
418 }
419 }
420}
421
422
423double
424MSAbstractLaneChangeModel::computeSpeedLat(double /*latDist*/, double& maneuverDist, bool /*urgent*/) const {
426 int stepsToChange = (int)ceil(fabs(maneuverDist) / SPEED2DIST(myVehicle.getVehicleType().getMaxSpeedLat()));
427 return DIST2SPEED(maneuverDist / stepsToChange);
428 } else {
429 return maneuverDist / STEPS2TIME(MSGlobals::gLaneChangeDuration);
430 }
431}
432
433
434double
438
439void
442 mySpeedLat = speedLat;
443}
444
445
446void
452
453
454bool
456 const bool pastBefore = pastMidpoint();
457 // maneuverDist is not updated in the context of continuous lane changing but represents the full LC distance
458 double maneuverDist = getManeuverDist();
459 setSpeedLat(computeSpeedLat(0, maneuverDist, (myOwnState & LCA_URGENT) != 0));
461 return !pastBefore && pastMidpoint();
462}
463
464
465void
467 UNUSED_PARAMETER(reason);
476 // opposite driving continues after parking
477 } else {
478 // aborted maneuver
479#ifdef DEBUG_OPPOSITE
480 if (debugVehicle()) {
481 std::cout << SIMTIME << " veh=" << myVehicle.getID() << " aborted maneuver (no longer opposite)\n";
482 }
483#endif
485 }
486 }
487}
488
489
490MSLane*
491MSAbstractLaneChangeModel::getShadowLane(const MSLane* lane, double posLat) const {
493 // initialize shadow lane
494 const double overlap = myVehicle.getLateralOverlap(posLat, lane);
495#ifdef DEBUG_SHADOWLANE
496 if (debugVehicle()) {
497 std::cout << SIMTIME << " veh=" << myVehicle.getID() << " posLat=" << posLat << " overlap=" << overlap << "\n";
498 }
499#endif
500 if (myAmOpposite) {
501 // return the neigh-lane in forward direction
502 return lane->getParallelLane(1);
503 } else if (overlap > NUMERICAL_EPS) {
504 const int shadowDirection = posLat < 0 ? -1 : 1;
505 return lane->getParallelLane(shadowDirection);
506 } else if (isChangingLanes() && myLaneChangeCompletion < 0.5) {
507 // "reserve" target lane even when there is no overlap yet
509 } else {
510 return nullptr;
511 }
512 } else {
513 return nullptr;
514 }
515}
516
517
518MSLane*
522
523
524void
526 if (myShadowLane != nullptr) {
527 if (debugVehicle()) {
528 std::cout << SIMTIME << " cleanupShadowLane\n";
529 }
531 myShadowLane = nullptr;
532 }
533 for (std::vector<MSLane*>::const_iterator it = myShadowFurtherLanes.begin(); it != myShadowFurtherLanes.end(); ++it) {
534 if (debugVehicle()) {
535 std::cout << SIMTIME << " cleanupShadowLane2\n";
536 }
538 }
539 myShadowFurtherLanes.clear();
541}
542
543void
545 if (myTargetLane != nullptr) {
546 if (debugVehicle()) {
547 std::cout << SIMTIME << " cleanupTargetLane\n";
548 }
550 myTargetLane = nullptr;
551 }
552 for (std::vector<MSLane*>::const_iterator it = myFurtherTargetLanes.begin(); it != myFurtherTargetLanes.end(); ++it) {
553 if (debugVehicle()) {
554 std::cout << SIMTIME << " cleanupTargetLane\n";
555 }
556 if (*it != nullptr) {
558 }
559 }
560 myFurtherTargetLanes.clear();
561// myNoPartiallyOccupatedByShadow.clear();
562}
563
564
565bool
567 // store request before canceling
568 getCanceledState(laneOffset) |= state;
569 int ret = myVehicle.influenceChangeDecision(state);
570 return ret != state;
571}
572
573double
577
578void
580 if (dir > 0) {
582 } else if (dir < 0) {
584 }
585}
586
587void
589 if (!MSGlobals::gSublane) {
590 // assume each vehicle drives at the center of its lane and act as if it fits
591 return;
592 }
593 if (myShadowLane != nullptr) {
594#ifdef DEBUG_SHADOWLANE
595 if (debugVehicle()) {
596 std::cout << SIMTIME << " updateShadowLane()\n";
597 }
598#endif
600 }
602 std::vector<MSLane*> passed;
603 if (myShadowLane != nullptr) {
605 const std::vector<MSLane*>& further = myVehicle.getFurtherLanes();
606 if (myAmOpposite) {
607 assert(further.size() == 0);
608 } else {
609 const std::vector<double>& furtherPosLat = myVehicle.getFurtherLanesPosLat();
610 assert(further.size() == furtherPosLat.size());
611 passed.push_back(myShadowLane);
612 for (int i = 0; i < (int)further.size(); ++i) {
613 MSLane* shadowFurther = getShadowLane(further[i], furtherPosLat[i]);
614#ifdef DEBUG_SHADOWLANE
615 if (debugVehicle()) {
616 std::cout << SIMTIME << " further=" << further[i]->getID() << " (posLat=" << furtherPosLat[i] << ") shadowFurther=" << Named::getIDSecure(shadowFurther) << "\n";
617 }
618#endif
619 if (shadowFurther != nullptr && shadowFurther->getLinkTo(passed.back()) != nullptr) {
620 passed.push_back(shadowFurther);
621 }
622 }
623 std::reverse(passed.begin(), passed.end());
624 }
625 } else {
626 if (isChangingLanes() && myVehicle.getLateralOverlap() > NUMERICAL_EPS) {
627 WRITE_WARNING("Vehicle '" + myVehicle.getID() + "' could not finish continuous lane change (lane disappeared) time=" +
628 time2string(MSNet::getInstance()->getCurrentTimeStep()) + ".");
630 }
631 }
632#ifdef DEBUG_SHADOWLANE
633 if (debugVehicle()) {
634 std::cout << SIMTIME << " updateShadowLane() veh=" << myVehicle.getID()
635 << " newShadowLane=" << Named::getIDSecure(myShadowLane)
636 << "\n before:" << " myShadowFurtherLanes=" << toString(myShadowFurtherLanes) << " further=" << toString(myVehicle.getFurtherLanes()) << " passed=" << toString(passed);
637 std::cout << std::endl;
638 }
639#endif
641#ifdef DEBUG_SHADOWLANE
642 if (debugVehicle()) std::cout
643 << "\n after:" << " myShadowFurtherLanes=" << toString(myShadowFurtherLanes) << "\n";
644#endif
645}
646
647
648int
650 if (isChangingLanes()) {
651 if (pastMidpoint()) {
652 return -myLaneChangeDirection;
653 } else {
655 }
656 } else if (myShadowLane == nullptr) {
657 return 0;
658 } else if (myAmOpposite) {
659 // return neigh-lane in forward direction
660 return 1;
661 } else if (&myShadowLane->getEdge() == &myVehicle.getLane()->getEdge()) {
663 } else {
664 // overlap with opposite direction lane
665 return 1;
666 }
667}
668
669
670MSLane*
672#ifdef DEBUG_TARGET_LANE
673 MSLane* oldTarget = myTargetLane;
674 std::vector<MSLane*> oldFurtherTargets = myFurtherTargetLanes;
675 if (debugVehicle()) {
676 std::cout << SIMTIME << " veh '" << myVehicle.getID() << "' (lane=" << myVehicle.getLane()->getID() << ") updateTargetLane()"
677 << "\n oldTarget: " << (oldTarget == nullptr ? "NULL" : oldTarget->getID())
678 << " oldFurtherTargets: " << toString(oldFurtherTargets);
679 }
680#endif
681 if (myTargetLane != nullptr) {
683 }
684 // Clear old further target lanes
685 for (MSLane* oldTargetLane : myFurtherTargetLanes) {
686 if (oldTargetLane != nullptr) {
687 oldTargetLane->resetManeuverReservation(&myVehicle);
688 }
689 }
690 myFurtherTargetLanes.clear();
691
692 // Get new target lanes and issue a maneuver reservation.
693 int targetDir;
695 if (myTargetLane != nullptr) {
697 // further targets are just the target lanes corresponding to the vehicle's further lanes
698 // @note In a neglectable amount of situations we might add a reservation for a shadow further lane.
699 for (MSLane* furtherLane : myVehicle.getFurtherLanes()) {
700 MSLane* furtherTargetLane = furtherLane->getParallelLane(targetDir);
701 myFurtherTargetLanes.push_back(furtherTargetLane);
702 if (furtherTargetLane != nullptr) {
703 furtherTargetLane->setManeuverReservation(&myVehicle);
704 }
705 }
706 }
707#ifdef DEBUG_TARGET_LANE
708 if (debugVehicle()) {
709 std::cout << "\n newTarget (maneuverDist=" << myManeuverDist << " offset=" << targetDir << "): " << (myTargetLane == nullptr ? "NULL" : myTargetLane->getID())
710 << " newFurtherTargets: " << toString(myFurtherTargetLanes)
711 << std::endl;
712 }
713#endif
714 return myTargetLane;
715}
716
717
718MSLane*
720 targetDir = 0;
721 if (myManeuverDist == 0) {
722 return nullptr;
723 }
724 // Current lateral boundaries of the vehicle
725 const double vehRight = myVehicle.getLateralPositionOnLane() - 0.5 * myVehicle.getWidth();
726 const double vehLeft = myVehicle.getLateralPositionOnLane() + 0.5 * myVehicle.getWidth();
727 const double halfLaneWidth = 0.5 * myVehicle.getLane()->getWidth();
728
729 if (vehRight + myManeuverDist < -halfLaneWidth) {
730 // Vehicle intends to traverse the right lane boundary
731 targetDir = -1;
732 } else if (vehLeft + myManeuverDist > halfLaneWidth) {
733 // Vehicle intends to traverse the left lane boundary
734 targetDir = 1;
735 }
736 if (targetDir == 0) {
737 // Presently, no maneuvering into another lane is begun.
738 return nullptr;
739 }
740 MSLane* target = myVehicle.getLane()->getParallelLane(targetDir);
741 if (target == nullptr || target == myShadowLane) {
742 return nullptr;
743 } else {
744 return target;
745 }
746}
747
748
749
750double
752 double result = 0.;
753 if (!(fabs(mySpeedLat) < NUMERICAL_EPS && fabs(myPreviousAngleOffset * 180 / M_PI) < NUMERICAL_EPS)) {
755 result = atan2(mySpeedLat, myVehicle.getSpeed());
756 } else {
758 }
759 }
760
761 myAngleOffset = result;
762 return result;
763}
764
765
766double
767MSAbstractLaneChangeModel::estimateLCDuration(const double speed, const double remainingManeuverDist, const double decel, bool urgent) const {
768
770 if (lcParams.find(SUMO_ATTR_LCA_MAXSPEEDLATSTANDING) == lcParams.end() && lcParams.find(SUMO_ATTR_LCA_MAXSPEEDLATFACTOR) == lcParams.end()) {
772 // no dependency of lateral speed on longitudinal speed. (Only called prior to LC initialization to determine whether it could be completed)
774 } else {
775 return remainingManeuverDist / myVehicle.getVehicleType().getMaxSpeedLat();
776 }
777 }
778
779 if (remainingManeuverDist == 0) {
780 return 0;
781 }
782
783 // Check argument assumptions
784 assert(speed >= 0);
785 assert(remainingManeuverDist >= 0);
786 assert(decel > 0);
789 assert(myMaxSpeedLatStanding >= 0);
790
791 // for brevity
792 const double v0 = speed;
793 const double D = remainingManeuverDist;
794 const double b = decel;
795 const double wmin = myMaxSpeedLatStanding;
796 const double f = myMaxSpeedLatFactor;
797 const double wmax = myVehicle.getVehicleType().getMaxSpeedLat();
798
799 /* Here's the approach for the calculation of the required time for the LC:
800 * To obtain the maximal LC-duration, for v(t) we assume that v(t)=max(0, v0-b*t),
801 * Where v(t)=0 <=> t >= ts:=v0/b
802 * For the lateral speed w(t) this gives:
803 * w(t) = min(wmax, wmin + f*v(t))
804 * The lateral distance covered until t is
805 * d(t) = int_0^t w(s) ds
806 * We distinguish three possibilities for the solution d(T)=D, where T is the time of the LC completion.
807 * 1) w(T) = wmax, i.e. v(T)>(wmax-wmin)/f
808 * 2) wmin < w(T) < wmax, i.e. (wmax-wmin)/f > v(T) > 0
809 * 3) w(T) = wmin, i.e., v(T)=0
810 */
811 const double vm = (wmax - wmin) / f;
812 double distSoFar = 0.;
813 double timeSoFar = 0.;
814 double v = v0;
815 if (v > vm) {
816 const double wmaxTime = (v0 - vm) / b;
817 const double d1 = wmax * wmaxTime;
818 if (d1 >= D) {
819 return D / wmax;
820 } else {
821 distSoFar += d1;
822 timeSoFar += wmaxTime;
823 v = vm;
824 }
825 }
826 if (v > 0) {
827 /* Here, w(t1+t) = wmin + f*v(t1+t) = wmin + f*(v - b*t)
828 * Thus, the additional lateral distance covered after time t is:
829 * d2 = (wmin + f*v)*t - 0.5*f*b*t^2
830 * and the additional lateral distance covered until v=0 at t=v/b is:
831 * d2 = (wmin + 0.5*f*v)*t
832 */
833 const double t = v / b; // stop time
834 const double d2 = (wmin + 0.5 * f * v) * t; // lateral distance covered until stop
835 assert(d2 > 0);
836 if (distSoFar + d2 >= D) {
837 // LC is completed during this phase
838 const double x = 0.5 * f * b;
839 const double y = wmin + f * v;
840 /* Solve D - distSoFar = y*t - x*t^2.
841 * 0 = x*t^2 - y*t/x + (D - distSoFar)/x
842 */
843 const double p = 0.5 * y / x;
844 const double q = (D - distSoFar) / x;
845 assert(p * p - q > 0);
846 const double t2 = p + sqrt(p * p - q);
847 return timeSoFar + t2;
848 } else {
849 distSoFar += d2;
850 timeSoFar += t;
851 //v = 0;
852 }
853 }
854 // If we didn't return yet this means the LC was not completed until the vehicle stops (if braking with rate b)
855 if (wmin == 0) {
856 // LC won't be completed if vehicle stands
857 double maneuverDist = remainingManeuverDist;
858 const double vModel = computeSpeedLat(maneuverDist, maneuverDist, urgent);
859 double result = D / vModel;
860 // make sure that the vehicle isn't braking to a stop during the manuever
861 if (vModel > SUMO_const_haltingSpeed && (vModel + myVehicle.getAcceleration() * result) > SUMO_const_haltingSpeed) {
862 // unless the model tells us something different
863 return result;
864 } else {
865 return -1;
866 }
867 } else {
868 // complete LC with lateral speed wmin
869 return timeSoFar + (D - distSoFar) / wmin;
870 }
871}
872
875 assert(isChangingLanes()); // Only to be called during ongoing lane change
877 if (lcParams.find(SUMO_ATTR_LCA_MAXSPEEDLATSTANDING) == lcParams.end() && lcParams.find(SUMO_ATTR_LCA_MAXSPEEDLATFACTOR) == lcParams.end()) {
880 } else {
882 }
883 }
884 // Using maxSpeedLat(Factor/Standing)
885 const bool urgent = (myOwnState & LCA_URGENT) != 0;
889}
890
891
892void
894 //std::cout << SIMTIME << " veh=" << myVehicle.getID() << " @=" << &myVehicle << " set shadow approaching=" << link->getViaLaneOrLane()->getID() << "\n";
895 myApproachedByShadow.push_back(link);
896}
897
898void
900 for (std::vector<MSLink*>::iterator it = myApproachedByShadow.begin(); it != myApproachedByShadow.end(); ++it) {
901 //std::cout << SIMTIME << " veh=" << myVehicle.getID() << " @=" << &myVehicle << " remove shadow approaching=" << (*it)->getViaLaneOrLane()->getID() << "\n";
902 (*it)->removeApproaching(&myVehicle);
903 }
904 myApproachedByShadow.clear();
905}
906
907
908
909void
912 int oldstate = myVehicle.getLaneChangeModel().getOwnState();
913 if (myOwnState != newstate) {
915 // Calculate and set the lateral maneuver distance corresponding to the change request
916 // to induce a corresponding sublane change.
917 const int dir = (newstate & LCA_RIGHT) != 0 ? -1 : ((newstate & LCA_LEFT) != 0 ? 1 : 0);
918 // minimum distance to move the vehicle fully onto the lane at offset dir
919 const double latLaneDist = myVehicle.lateralDistanceToLane(dir);
920 if ((newstate & LCA_TRACI) != 0) {
921 if ((newstate & LCA_STAY) != 0) {
922 setManeuverDist(0.);
923 } else if (((newstate & LCA_RIGHT) != 0 && dir < 0)
924 || ((newstate & LCA_LEFT) != 0 && dir > 0)) {
925 setManeuverDist(latLaneDist);
926 }
927 }
928 if (myVehicle.hasInfluencer()) {
929 // lane change requests override sublane change requests
931 }
932
933 }
934 setOwnState(newstate);
935 } else {
936 // Check for sublane change requests
938 const double maneuverDist = myVehicle.getInfluencer().getLatDist();
941 newstate |= LCA_TRACI;
942 if (myOwnState != newstate) {
943 setOwnState(newstate);
944 }
945 if (gDebugFlag2) {
946 std::cout << " traci influenced maneuverDist=" << maneuverDist << "\n";
947 }
948 }
949 }
950 if (gDebugFlag2) {
951 std::cout << SIMTIME << " veh=" << myVehicle.getID() << " stateAfterTraCI=" << toString((LaneChangeAction)newstate) << " original=" << toString((LaneChangeAction)oldstate) << "\n";
952 }
953}
954
955void
960
961void
963 if (follower.first != 0) {
964 myLastFollowerGap = follower.second + follower.first->getVehicleType().getMinGap();
966 myLastFollowerSpeed = follower.first->getSpeed();
967 }
968}
969
970void
972 if (leader.first != 0) {
974 myLastLeaderSecureGap = secGap;
975 myLastLeaderSpeed = leader.first->getSpeed();
976 }
977}
978
979void
981 if (leader.first != 0) {
984 myLastOrigLeaderSpeed = leader.first->getSpeed();
985 }
986}
987
988void
1011
1012void
1014 int rightmost;
1015 int leftmost;
1016 vehicles.getSubLanes(&myVehicle, 0, rightmost, leftmost);
1017 for (int i = rightmost; i <= leftmost; ++i) {
1018 CLeaderDist vehDist = vehicles[i];
1019 if (vehDist.first != 0) {
1020 const MSVehicle* leader = &myVehicle;
1021 const MSVehicle* follower = vehDist.first;
1022 const double netGap = vehDist.second + follower->getVehicleType().getMinGap();
1023 if (netGap < myLastFollowerGap && netGap >= 0) {
1024 myLastFollowerGap = netGap;
1025 myLastFollowerSecureGap = follower->getCarFollowModel().getSecureGap(follower, leader, follower->getSpeed(), leader->getSpeed(), leader->getCarFollowModel().getMaxDecel());
1026 myLastFollowerSpeed = follower->getSpeed();
1027 }
1028 }
1029 }
1030}
1031
1032void
1034 int rightmost;
1035 int leftmost;
1036 vehicles.getSubLanes(&myVehicle, 0, rightmost, leftmost);
1037 for (int i = rightmost; i <= leftmost; ++i) {
1038 CLeaderDist vehDist = vehicles[i];
1039 if (vehDist.first != 0) {
1040 const MSVehicle* leader = vehDist.first;
1041 const MSVehicle* follower = &myVehicle;
1042 const double netGap = vehDist.second + follower->getVehicleType().getMinGap();
1043 if (netGap < myLastLeaderGap && netGap >= 0) {
1044 myLastLeaderGap = netGap;
1045 myLastLeaderSecureGap = follower->getCarFollowModel().getSecureGap(follower, leader, follower->getSpeed(), leader->getSpeed(), leader->getCarFollowModel().getMaxDecel());
1046 myLastLeaderSpeed = leader->getSpeed();
1047 }
1048 }
1049 }
1050}
1051
1052void
1054 int rightmost;
1055 int leftmost;
1056 vehicles.getSubLanes(&myVehicle, 0, rightmost, leftmost);
1057 for (int i = rightmost; i <= leftmost; ++i) {
1058 CLeaderDist vehDist = vehicles[i];
1059 if (vehDist.first != 0) {
1060 const MSVehicle* leader = vehDist.first;
1061 const MSVehicle* follower = &myVehicle;
1062 const double netGap = vehDist.second + follower->getVehicleType().getMinGap();
1063 if (netGap < myLastOrigLeaderGap && netGap >= 0) {
1064 myLastOrigLeaderGap = netGap;
1065 myLastOrigLeaderSecureGap = follower->getCarFollowModel().getSecureGap(follower, leader, follower->getSpeed(), leader->getSpeed(), leader->getCarFollowModel().getMaxDecel());
1066 myLastOrigLeaderSpeed = leader->getSpeed();
1067 }
1068 }
1069 }
1070}
1071
1072
1073bool
1075 const int stateRight = mySavedStateRight.second;
1076 if (
1077 (stateRight & LCA_STRATEGIC) != 0
1078 && (stateRight & LCA_RIGHT) != 0
1079 && (stateRight & LCA_BLOCKED) != 0) {
1080 return true;
1081 }
1082 const int stateLeft = mySavedStateLeft.second;
1083 if (
1084 (stateLeft & LCA_STRATEGIC) != 0
1085 && (stateLeft & LCA_LEFT) != 0
1086 && (stateLeft & LCA_BLOCKED) != 0) {
1087 return true;
1088 }
1089 return false;
1090}
1091
1092double
1096
1097
1098int
1100 const int i = myVehicle.getLane()->getIndex();
1101 if (myAmOpposite) {
1103 } else {
1104 return i;
1105 }
1106}
1107
1108void
1109MSAbstractLaneChangeModel::addLCSpeedAdvice(const double vSafe, bool ownAdvice) {
1110 const double accel = SPEED2ACCEL(vSafe - myVehicle.getSpeed());
1111 myLCAccelerationAdvices.push_back({accel, ownAdvice});
1112}
1113
1114
1115bool
1116MSAbstractLaneChangeModel::canOvertakeRight(const MSVehicle* const nv, const double dist, const double maxSpeedDiff, const double helpOvertakeSpeed, double& vSafe, double& deltaV) const {
1117 deltaV = MAX2(maxSpeedDiff, myVehicle.getSpeed() - nv->getSpeed());
1118 if (deltaV > 0) {
1119 const double vMaxDecel = getCarFollowModel().getSpeedAfterMaxDecel(myVehicle.getSpeed());
1120 const double vSafeFollow = getCarFollowModel().followSpeed(
1122 const double vStayBehind = nv->getSpeed() - helpOvertakeSpeed;
1123 if (vSafeFollow >= vMaxDecel) {
1124 vSafe = vSafeFollow;
1125 } else {
1126 vSafe = MAX2(vMaxDecel, vStayBehind);
1127 }
1128 return true;
1129 }
1130 return false;
1131}
1132
1133
1134void
1136 std::vector<double> lcState;
1137 lcState.push_back((double)myOwnState);
1138 for (const auto& item : myLCAccelerationAdvices) {
1139 lcState.push_back(item.first);
1140 lcState.push_back((double)item.second);
1141 }
1142 out.writeAttr(SUMO_ATTR_LCSTATE_BASE, lcState);
1143
1146 }
1147}
1148
1149void
1152 std::istringstream bis(attrs.getString(SUMO_ATTR_LCSTATE_BASE));
1153 double token;
1154 bis >> token;
1155 myOwnState = (int)token; // double is suffciently precise
1156 double prev = std::numeric_limits<double>::max();
1157 while (bis >> token) {
1158 if (prev != std::numeric_limits<double>::max()) {
1159 myLCAccelerationAdvices.push_back(std::make_pair(prev, (bool)token));
1160 prev = std::numeric_limits<double>::max();
1161 }
1162 prev = token;
1163 }
1164 }
1165 if (attrs.hasAttribute(SUMO_ATTR_LCSTATE)) {
1166 std::istringstream bis(attrs.getString(SUMO_ATTR_LCSTATE));
1167 bis >> mySpeedLat;
1169 bis >> myLaneChangeDirection;
1170 }
1171}
1172
1173
1174double
1175MSAbstractLaneChangeModel::getExtraReservation(int bestLaneOffset, double neighExtraDist) const {
1176 if (neighExtraDist > myVehicle.getVehicleType().getLengthWithGap()) {
1177 return 0;
1178 }
1179 if (bestLaneOffset < -1) {
1180 return 20;
1181 } else if (bestLaneOffset > 1) {
1182 return 40;
1183 }
1184 return 0;
1185}
1186
1187
1188double
1189MSAbstractLaneChangeModel::getCooperativeHelpSpeed(const MSLane* lane, double distToLaneEnd) const {
1190 if (myCooperativeHelpTime >= 0) {
1191 std::pair<double, SUMOTime> backAndWaiting = lane->getEdge().getLastBlocked(lane->getIndex());
1192 if (backAndWaiting.second >= myCooperativeHelpTime) {
1193 double gap = distToLaneEnd - lane->getLength() + backAndWaiting.first - myVehicle.getVehicleType().getMinGap() - NUMERICAL_EPS;
1194 if (backAndWaiting.first < 0) {
1195 if (myVehicle.getLane()->getToJunction() == lane->getFromJunction()) {
1196 if (myVehicle.getLane()->isInternal()) {
1197 // already on the junction, potentially blocking lane change, do not stop
1198 gap = -1;
1199 } else {
1200 // stop before entering the junction
1202 }
1203 }
1204 }
1205 if (gap > 0) {
1206 double stopSpeed = myVehicle.getCarFollowModel().stopSpeed(&myVehicle, myVehicle.getSpeed(), gap);
1207 //if (myVehicle.isSelected() && stopSpeed < myVehicle.getSpeed()) {
1208 // std::cout << SIMTIME << " veh=" << myVehicle.getID() << " lane=" << lane->getID() << " dte=" << distToLaneEnd << " gap=" << gap << " backPos=" << backAndWaiting.first << " waiting=" << backAndWaiting.second << " helpTime=" << myCooperativeHelpTime << " stopSpeed=" << stopSpeed << " minNext=" << myVehicle.getCarFollowModel().minNextSpeed(myVehicle.getSpeed(), &myVehicle) << "\n";
1209 //}
1211 // regular braking is helpful
1212 return stopSpeed;
1213 }
1214 }
1215 }
1216 }
1217 // do not restrict speed
1218 return std::numeric_limits<double>::max();
1219}
long long int SUMOTime
Definition GUI.h:36
#define LC_ASSUMED_DECEL
std::pair< const MSVehicle *, double > CLeaderDist
#define WRITE_WARNING(msg)
Definition MsgHandler.h:286
#define TLF(string,...)
Definition MsgHandler.h:306
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 STEPS2TIME(x)
Definition SUMOTime.h:58
#define SPEED2DIST(x)
Definition SUMOTime.h:48
#define SIMTIME
Definition SUMOTime.h:65
#define TIME2STEPS(x)
Definition SUMOTime.h:60
#define DIST2SPEED(x)
Definition SUMOTime.h:50
#define SPEED2ACCEL(x)
Definition SUMOTime.h:56
const long long int VTYPEPARS_MAXSPEED_LAT_SET
@ SVC_BICYCLE
vehicle is a bicycle
@ SVC_MOTORCYCLE
vehicle is a motorcycle
@ SVC_EMERGENCY
public emergency vehicles
@ SVC_MOPED
vehicle is a moped
LaneChangeAction
The state of a vehicle's lane-change behavior.
@ LCA_UNKNOWN
The action has not been determined.
@ LCA_BLOCKED
blocked in all directions
@ LCA_URGENT
The action is urgent (to be defined by lc-model)
@ LCA_STAY
Needs to stay on the current lane.
@ LCA_AMBACKBLOCKER
@ LCA_AMBLOCKINGLEADER
@ LCA_LEFT
Wants go to the left.
@ LCA_STRATEGIC
The action is needed to follow the route (navigational lc)
@ LCA_AMBACKBLOCKER_STANDING
@ LCA_TRACI
The action is due to a TraCI request.
@ LCA_RIGHT
Wants go to the right.
@ LCA_AMBLOCKINGFOLLOWER
@ SUMO_ATTR_SPEED
@ SUMO_ATTR_LCA_ASSERTIVE
@ SUMO_ATTR_Y
@ SUMO_ATTR_X
@ SUMO_ATTR_LCA_MAXDISTLATSTANDING
@ SUMO_ATTR_LCA_COOPERATIVE_HELPTIME
@ SUMO_ATTR_LCA_MAXSPEEDLATFACTOR
@ SUMO_ATTR_LCA_MAXSPEEDLATSTANDING
@ SUMO_ATTR_LCA_STRATEGIC_LOOKAHEAD
@ SUMO_ATTR_TO
@ SUMO_ATTR_FROM
@ SUMO_ATTR_LCA_SIGMA
@ SUMO_ATTR_LCSTATE_BASE
@ SUMO_ATTR_TYPE
@ SUMO_ATTR_ID
@ SUMO_ATTR_LCSTATE
The state of the lanechange model.
@ SUMO_ATTR_LCA_OVERTAKE_RIGHT
@ SUMO_ATTR_DIR
The abstract direction of a link.
@ SUMO_ATTR_POSITION
@ SUMO_ATTR_TIME
trigger: the time of the step
bool gDebugFlag2
Definition StdDefs.cpp:45
const double SUMO_const_haltingSpeed
the speed threshold at which vehicles are considered as halting
Definition StdDefs.h:62
T MAX2(T a, T b)
Definition StdDefs.h:86
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:49
Interface for lane-change models.
double getForwardPos() const
get vehicle position relative to the forward direction lane
double myAccelerationLat
the current lateral acceleration
void setFollowerGaps(CLeaderDist follower, double secGap)
virtual double getExtraReservation(int bestLaneOffset, double neighExtraDist=0) const
std::vector< MSLane * > myFurtherTargetLanes
bool myAlreadyChanged
whether the vehicle has already moved this step
bool myAmOpposite
whether the vehicle is driving in the opposite direction
std::shared_ptr< MSLeaderDistanceInfo > myRightFollowers
std::shared_ptr< MSLeaderDistanceInfo > myRightLeaders
virtual void setOwnState(const int state)
bool pastMidpoint() const
return whether the vehicle passed the midpoint of a continuous lane change maneuver
double myPreviousAngleOffset
the angle offset of the previous time step resulting from lane change and sigma
virtual double getAssumedDecelForLaneChangeDuration() const
Returns a deceleration value which is used for the estimation of the duration of a lane change.
virtual double computeSpeedLat(double latDist, double &maneuverDist, bool urgent) const
decides the next lateral speed depending on the remaining lane change distance to be covered and upda...
virtual double estimateLCDuration(const double speed, const double remainingManeuverDist, const double decel, bool urgent) const
Calculates the maximal time needed to complete a lane change maneuver if lcMaxSpeedLatFactor and lcMa...
std::shared_ptr< MSLeaderDistanceInfo > myLeftLeaders
int myPreviousState
lane changing state from the previous simulation step
double getManeuverDist() const
Returns the remaining unblocked distance for the current maneuver. (only used by sublane model)
int myOwnState
The current state of the vehicle.
double myLastOrigLeaderGap
acutal and secure distance to closest leader vehicle on the original when performing lane change
virtual bool predInteraction(const std::pair< MSVehicle *, double > &leader)
void laneChangeOutput(const std::string &tag, MSLane *source, MSLane *target, int direction, double maneuverDist=0)
called once the vehicle ends a lane change manoeuvre (non-instant)
bool myDontResetLCGaps
Flag to prevent resetting the memorized values for LC relevant gaps until the LC output is triggered ...
int myPreviousState2
lane changing state from step before the previous simulation step
MSLane * getShadowLane() const
Returns the lane the vehicle's shadow is on during continuous/sublane lane change.
const std::shared_ptr< MSLeaderDistanceInfo > getFollowers(const int dir)
Returns the neighboring, lc-relevant followers for the last step in the requested direction.
double myCommittedSpeed
the speed when committing to a change maneuver
std::shared_ptr< MSLeaderDistanceInfo > myLeftFollowers
Cached info on lc-relevant neighboring vehicles.
static bool myLCOutput
whether to record lane-changing
bool startLaneChangeManeuver(MSLane *source, MSLane *target, int direction)
start the lane change maneuver and return whether it continues
virtual void saveState(OutputDevice &out) const
Save the state of the laneChangeModel.
std::pair< int, int > mySavedStateRight
double myLastLeaderSecureGap
the minimum longitudinal distances to vehicles on the target lane that would be necessary for stringe...
void endLaneChangeManeuver(const MSMoveReminder::Notification reason=MSMoveReminder::NOTIFICATION_LANE_CHANGE)
void saveLCState(const int dir, int stateWithoutTraCI, const int state)
static bool myAllowOvertakingRight
whether overtaking on the right is permitted
std::vector< MSLink * > myApproachedByShadow
links which are approached by the shadow vehicle
bool canOvertakeRight(const MSVehicle *const nv, const double dist, const double maxSpeedDiff, const double helpOvertakeSpeed, double &vSafe, double &deltaV) const
void addLCSpeedAdvice(const double vSafe, bool ownAdvice=true)
Takes a vSafe (speed advice for speed in the next simulation step), converts it into an acceleration ...
void setLeaderGaps(CLeaderDist, double secGap)
const std::shared_ptr< MSLeaderDistanceInfo > getLeaders(const int dir)
Returns the neighboring, lc-relevant leaders for the last step in the requested direction.
std::vector< MSLane * > myNoPartiallyOccupatedByShadow
bool cancelRequest(int state, int laneOffset)
whether the influencer cancels the given request
double myLastLeaderGap
the actual minimum longitudinal distances to vehicles on the target lane
SUMOTime remainingTime() const
Compute the remaining time until LC completion.
void setOrigLeaderGaps(CLeaderDist, double secGap)
void setManeuverDist(const double dist)
Updates the remaining distance for the current maneuver while it is continued within non-action steps...
std::vector< std::pair< double, bool > > myLCAccelerationAdvices
void setShadowApproachingInformation(MSLink *link) const
set approach information for the shadow vehicle
double getCooperativeHelpSpeed(const MSLane *lane, double distToLaneEnd) const
return speed for helping a vehicle that is blocked from changing
virtual bool avoidOvertakeRight(const MSVehicle *const neighLeader, const bool allowProb=false) const
int getNormalizedLaneIndex()
brief return lane index that treats opposite lanes like normal lanes to the left of the forward lanes
static MSAbstractLaneChangeModel * build(LaneChangeModel lcm, MSVehicle &vehicle)
Factory method for instantiating new lane changing models.
void changedToOpposite()
called when a vehicle changes between lanes in opposite directions
void setSpeedLat(double speedLat)
set the lateral speed and update lateral acceleraton
MSLane * myTargetLane
The target lane for the vehicle's current maneuver.
MSLane * determineTargetLane(int &targetDir) const
double myPreviousManeuverDist
Maneuver distance from the previous simulation step.
double getMaxSpeedLat2() const
return the max of maxSpeedLat and lcMaxSpeedLatStanding
std::vector< double > myShadowFurtherLanesPosLat
const MSCFModel & getCarFollowModel() const
The vehicle's car following model.
MSLane * myShadowLane
A lane that is partially occupied by the front of the vehicle but that is not the primary lane.
double mySpeedLat
the current lateral speed
virtual void updateSafeLatDist(const double travelledLatDist)
Updates the value of safe lateral distances (in SL2015) during maneuver continuation in non-action st...
void checkTraCICommands()
Check for commands issued for the vehicle via TraCI and apply the appropriate state changes For the s...
double myManeuverDist
The complete lateral distance the vehicle wants to travel to finish its maneuver Only used by sublane...
int myLaneChangeDirection
direction of the lane change maneuver -1 means right, 1 means left
void primaryLaneChanged(MSLane *source, MSLane *target, int direction)
called once when the vehicles primary lane changes
int getShadowDirection() const
return the direction in which the current shadow lane lies
double myLastLeaderSpeed
speeds of surrounding vehicles at the time of lane change
virtual void loadState(const SUMOSAXAttributes &attrs)
Loads the state of the laneChangeModel from the given attributes.
MSAbstractLaneChangeModel(MSVehicle &v, const LaneChangeModel model)
Constructor.
MSVehicle & myVehicle
The vehicle this lane-changer belongs to.
double calcAngleOffset()
return the angle offset during a continuous change maneuver
double myAngleOffset
the current angle offset resulting from lane change and sigma
double myLastLateralGapLeft
the minimum lateral gaps to other vehicles that were found when last changing to the left and right
virtual ~MSAbstractLaneChangeModel()
Destructor.
static void initGlobalOptions(const OptionsCont &oc)
init global model parameters
void memorizeGapsAtLCInit()
Control for resetting the memorized values for LC relevant gaps until the LC output is triggered in t...
double myLaneChangeCompletion
progress of the lane change maneuver 0:started, 1:complete
virtual bool debugVehicle() const
whether the current vehicles shall be debugged
virtual void changed()=0
bool isChangingLanes() const
return true if the vehicle currently performs a lane change maneuver
std::vector< MSLane * > myShadowFurtherLanes
virtual bool congested(const MSVehicle *const neighLeader)
void clearNeighbors()
Clear info on neighboring vehicle from previous step.
void saveNeighbors(const int dir, const MSLeaderDistanceInfo &followers, const MSLeaderDistanceInfo &leaders)
Saves the lane change relevant vehicles, which are currently on neighboring lanes in the given direct...
const SUMOVehicleParameter & getParameter() const
Returns the vehicle's parameter (including departure definition)
double getLength() const
Returns the vehicle's length.
double getWidth() const
Returns the vehicle's width.
const std::list< MSStop > & getStops() const
SumoRNG * getRNG() const
const MSStop & getNextStop() const
const MSVehicleType & getVehicleType() const
Returns the vehicle's type definition.
bool isStopped() const
Returns whether the vehicle is at a stop.
virtual double interactionGap(const MSVehicle *const veh, double vL) const
Returns the maximum gap at which an interaction between both vehicles occurs.
virtual double minNextSpeed(double speed, const MSVehicle *const veh=0) const
Returns the minimum speed given the current speed (depends on the numerical update scheme and its ste...
virtual double getSecureGap(const MSVehicle *const veh, const MSVehicle *const, const double speed, const double leaderSpeed, const double leaderMaxDecel) const
Returns the minimum gap to reserve if the leader is braking at maximum (>=0)
double getMaxDecel() const
Get the vehicle type's maximal comfortable deceleration [m/s^2].
Definition MSCFModel.h:269
virtual double getSpeedAfterMaxDecel(double v) const
Returns the velocity after maximum deceleration.
Definition MSCFModel.h:431
virtual double followSpeed(const MSVehicle *const veh, double speed, double gap2pred, double predSpeed, double predMaxDecel, const MSVehicle *const pred=0, const CalcReason usage=CalcReason::CURRENT) const =0
Computes the vehicle's follow speed (no dawdling)
double stopSpeed(const MSVehicle *const veh, const double speed, double gap, const CalcReason usage=CalcReason::CURRENT) const
Computes the vehicle's safe speed for approaching a non-moving obstacle (no dawdling)
Definition MSCFModel.h:173
A device which collects info on the vehicle trip (mainly on departure and arrival)
std::pair< double, SUMOTime > getLastBlocked(int index) const
retrieve properties of a blocked vehicle that wants to chane to the lane with the given index
Definition MSEdge.cpp:1711
int getNumLanes() const
Definition MSEdge.h:172
static double gLateralResolution
Definition MSGlobals.h:100
static bool gLefthand
Whether lefthand-drive is being simulated.
Definition MSGlobals.h:177
static bool gSublane
whether sublane simulation is enabled (sublane model or continuous lanechanging)
Definition MSGlobals.h:168
static SUMOTime gLaneChangeDuration
Definition MSGlobals.h:97
A lane change model developed by D. Krajzewicz between 2004 and 2010.
A lane change model developed by D. Krajzewicz, J. Erdmann et al. between 2004 and 2013,...
A lane change model developed by D. Krajzewicz, J. Erdmann et al. between 2004 and 2013.
A lane change model developed by J. Erdmann.
Representation of a lane in the micro simulation.
Definition MSLane.h:84
MSLane * getParallelLane(int offset, bool includeOpposite=true) const
Returns the lane with the given offset parallel to this one or 0 if it does not exist.
Definition MSLane.cpp:2887
virtual void resetManeuverReservation(MSVehicle *v)
Unregisters a vehicle, which previously registered for maneuvering into this lane.
Definition MSLane.cpp:437
const MSLink * getLinkTo(const MSLane *const) const
returns the link to the given lane or nullptr, if it is not connected
Definition MSLane.cpp:2783
void forceVehicleInsertion(MSVehicle *veh, double pos, MSMoveReminder::Notification notification, double posLat=0)
Inserts the given vehicle at the given position.
Definition MSLane.cpp:1423
double getSpeedLimit() const
Returns the lane's maximum allowed speed.
Definition MSLane.h:603
void enteredByLaneChange(MSVehicle *v)
Definition MSLane.cpp:3396
const MSJunction * getToJunction() const
Definition MSLane.cpp:4766
double getLength() const
Returns the lane's length.
Definition MSLane.h:617
const MSJunction * getFromJunction() const
Definition MSLane.cpp:4760
virtual double setPartialOccupation(MSVehicle *v)
Sets the information about a vehicle lapping into this lane.
Definition MSLane.cpp:384
int getIndex() const
Returns the lane's index.
Definition MSLane.h:653
void leftByLaneChange(MSVehicle *v)
Definition MSLane.cpp:3389
double getOppositePos(double pos) const
return the corresponding position on the opposite lane
Definition MSLane.cpp:4431
bool isInternal() const
Definition MSLane.cpp:2651
virtual void resetPartialOccupation(MSVehicle *v)
Removes the information about a vehicle lapping into this lane.
Definition MSLane.cpp:403
virtual void setManeuverReservation(MSVehicle *v)
Registers the lane change intentions (towards this lane) for the given vehicle.
Definition MSLane.cpp:426
MSLane * getParallelOpposite() const
return the opposite direction lane of this lanes edge or nullptr
Definition MSLane.cpp:4425
MSEdge & getEdge() const
Returns the lane's edge.
Definition MSLane.h:775
double getWidth() const
Returns the lane's width.
Definition MSLane.h:646
saves leader/follower vehicles and their distances relative to an ego vehicle
void getSubLanes(const MSVehicle *veh, double latOffset, int &rightmost, int &leftmost) const
Notification
Definition of a vehicle state.
@ NOTIFICATION_LANE_CHANGE
The vehicle changes lanes (micro only)
@ NOTIFICATION_PARKING
The vehicle starts or ends parking.
static MSNet * getInstance()
Returns the pointer to the unique instance of MSNet (singleton).
Definition MSNet.cpp:199
bool isOpposite
whether this an opposite-direction stop
Definition MSStop.h:87
double getLatDist() const
Definition MSVehicle.h:1603
Representation of a vehicle in the micro simulation.
Definition MSVehicle.h:77
const std::vector< double > & getFurtherLanesPosLat() const
Definition MSVehicle.h:839
void setTentativeLaneAndPosition(MSLane *lane, double pos, double posLat=0)
set tentative lane and position during insertion to ensure that all cfmodels work (some of them requi...
MSAbstractLaneChangeModel & getLaneChangeModel()
int influenceChangeDecision(int state)
allow TraCI to influence a lane change decision
void fixPosition()
repair errors in vehicle position after changing between internal edges
double getAcceleration() const
Returns the vehicle's acceleration in m/s (this is computed as the last step's mean acceleration in c...
Definition MSVehicle.h:514
Position getPosition(const double offset=0) const
Return current position (x/y, cartesian)
double lateralDistanceToLane(const int offset) const
Get the minimal lateral distance required to move fully onto the lane at given offset.
void leaveLane(const MSMoveReminder::Notification reason, const MSLane *approachedLane=0)
Update of members if vehicle leaves a new lane in the lane change step or at arrival.
void switchOffSignal(int signal)
Switches the given signal off.
Definition MSVehicle.h:1173
@ VEH_SIGNAL_BLINKER_RIGHT
Right blinker lights are switched on.
Definition MSVehicle.h:1112
@ VEH_SIGNAL_BLINKER_LEFT
Left blinker lights are switched on.
Definition MSVehicle.h:1114
const MSLane * getLane() const
Returns the lane the vehicle is on.
Definition MSVehicle.h:581
void enterLaneAtLaneChange(MSLane *enteredLane)
Update when the vehicle enters a new lane in the laneChange step.
Influencer & getInfluencer()
double updateFurtherLanes(std::vector< MSLane * > &furtherLanes, std::vector< double > &furtherLanesPosLat, const std::vector< MSLane * > &passedLanes)
update a vector of further lanes and return the new backPos
double getLateralPositionOnLane() const
Get the vehicle's lateral position on the lane.
Definition MSVehicle.h:413
bool congested() const
double getSpeed() const
Returns the vehicle's current speed.
Definition MSVehicle.h:490
const std::vector< MSLane * > & getFurtherLanes() const
Definition MSVehicle.h:835
const MSCFModel & getCarFollowModel() const
Returns the vehicle's car following model definition.
Definition MSVehicle.h:969
double getPositionOnLane() const
Get the vehicle's position along the lane.
Definition MSVehicle.h:374
double getLateralOverlap() const
return the amount by which the vehicle extends laterally outside it's primary lane
bool hasInfluencer() const
whether the vehicle is individually influenced (via TraCI or special parameters)
Definition MSVehicle.h:1702
void switchOnSignal(int signal)
Switches the given signal on.
Definition MSVehicle.h:1165
void updateDriveItems()
Check whether the drive items (myLFLinkLanes) are up to date, and update them if required.
double getLengthWithGap() const
Get vehicle's length including the minimum gap [m].
SUMOVehicleClass getVehicleClass() const
Get this vehicle type's vehicle class.
const std::string & getID() const
Returns the name of the vehicle type.
double getMinGap() const
Get the free space in front of vehicles of this class.
bool wasSet(long long int what) const
Returns whether the given parameter was set.
double getMaxSpeedLat() const
Get vehicle's maximum lateral speed [m/s].
const SUMOVTypeParameter & getParameter() const
static std::string getIDSecure(const T *obj, const std::string &fallBack="NULL")
get an identifier for Named-like object which may be Null
Definition Named.h:67
const std::string & getID() const
Returns the id.
Definition Named.h:74
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.
bool getBool(const std::string &name) const
Returns the boolean-value of the named option (only for Option_Bool)
Static storage of an output device and its base (abstract) implementation.
OutputDevice & openTag(const std::string &xmlElement)
Opens an XML tag.
OutputDevice & writeAttr(const ATTR_TYPE &attr, const T &val, const bool isNull=false)
writes a named attribute
static OutputDevice & getDeviceByOption(const std::string &name)
Returns the device described by the option.
bool closeTag(const std::string &comment="")
Closes the most recently opened tag and optionally adds a comment.
virtual const std::string getParameter(const std::string &key, const std::string defaultValue="") const
Returns the value for a given key.
double x() const
Returns the x-position.
Definition Position.h:52
double y() const
Returns the y-position.
Definition Position.h:57
static double rand(SumoRNG *rng=nullptr)
Returns a random real number in [0, 1)
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.
virtual bool hasAttribute(int id) const =0
Returns the information whether the named (by its enum-value) attribute is within the current list.
std::map< SumoXMLAttr, std::string > SubParams
sub-model parameters
const SubParams & getLCParams() const
Returns the LC parameter.
static bool toBool(const std::string &sData)
converts a string into the bool value described by it by calling the char-type converter
#define UNUSED_PARAMETER(x)
#define DEBUG_COND
Definition json.hpp:4471
#define M_PI
Definition odrSpiral.cpp:45