41#define LOOK_FORWARD 10.
45#define LCA_RIGHT_IMPATIENCE -1.
46#define CUT_IN_LEFT_SPEED_THRESHOLD 27.
47#define MAX_ONRAMP_LENGTH 200.
49#define LOOK_AHEAD_MIN_SPEED 0.0
50#define LOOK_AHEAD_SPEED_MEMORY 0.9
52#define HELP_DECEL_FACTOR 1.0
54#define HELP_OVERTAKE (10.0 / 3.6)
55#define MIN_FALLBEHIND (7.0 / 3.6)
59#define KEEP_RIGHT_TIME 5.0
61#define RELGAIN_NORMALIZATION_MIN_SPEED 10.0
63#define TURN_LANE_DIST 200.0
64#define GAIN_PERCEPTION_THRESHOLD 0.05
66#define ARRIVALPOS_LAT_THRESHOLD 100.0
69#define LATGAP_SPEED_THRESHOLD (50 / 3.6)
72#define LATGAP_SPEED_THRESHOLD2 (50 / 3.6)
75#define SPEEDGAIN_DECAY_FACTOR 0.5
77#define SPEEDGAIN_MEMORY_FACTOR 0.5
79#define REACT_TO_STOPPED_DISTANCE 100
105#define DEBUG_COND (myVehicle.isSelected())
117 mySpeedGainProbabilityRight(0),
118 mySpeedGainProbabilityLeft(0),
119 myKeepRightProbability(0),
120 myLeadingBlockerLength(0),
124 myCanChangeFully(true),
125 mySafeLatDistRight(0),
126 mySafeLatDistLeft(0),
134 myMinGapLat(v.getVehicleType().getMinGapLat()),
137 MAX2(NUMERICAL_EPS, myMinGapLat)) /
138 MAX2(NUMERICAL_EPS, myMinGapLat)))),
140 myMinImpatience(myImpatience),
192 const std::vector<MSVehicle::LaneQ>& preb,
195 double& latDist,
double& maneuverDist,
int& blocked) {
198 const std::string changeType = laneOffset == -1 ?
"right" : (laneOffset == 1 ?
"left" :
"current");
206 <<
" neigh=" << neighLane.
getID()
211 <<
" considerChangeTo=" << changeType
218 leaders, followers, blockers,
219 neighLeaders, neighFollowers, neighBlockers,
221 lastBlocked, firstBlocked, latDist, maneuverDist, blocked);
223 result =
keepLatGap(result, leaders, followers, blockers,
224 neighLeaders, neighFollowers, neighBlockers,
225 neighLane, laneOffset, latDist, maneuverDist, blocked);
227 result |=
getLCA(result, latDist);
229#if defined(DEBUG_MANEUVER) || defined(DEBUG_STATE)
230 double latDistTmp = latDist;
233#if defined(DEBUG_MANEUVER) || defined(DEBUG_STATE)
235 std::cout <<
SIMTIME <<
" veh=" <<
myVehicle.
getID() <<
" maneuverDist=" << maneuverDist <<
" latDist=" << latDistTmp <<
" mySpeedPrev=" <<
mySpeedLat <<
" speedLat=" <<
DIST2SPEED(latDist) <<
" latDist2=" << latDist <<
"\n";
242 <<
" wantsChangeTo=" << changeType
243 <<
" latDist=" << latDist
244 <<
" maneuverDist=" << maneuverDist
252 <<
" wantsNoChangeTo=" << changeType
309 const double newSpeed =
_patchSpeed(
MAX2(min, 0.0), wanted, max, cfModel);
310#ifdef DEBUG_PATCHSPEED
312 const std::string patched = (wanted != newSpeed ?
" patched=" +
toString(newSpeed) :
"");
319 <<
" wanted=" << wanted
338 double nVSafe = wanted;
344#ifdef DEBUG_PATCHSPEED
352 max =
MIN2(max, safe);
357 if (safe >= vMinEmergency) {
359 min =
MAX2(vMinEmergency, safe);
362#ifdef DEBUG_PATCHSPEED
364 std::cout <<
SIMTIME <<
" veh=" <<
myVehicle.
getID() <<
" slowing down for leading blocker, safe=" << safe << (safe + NUMERICAL_EPS < min ?
" (not enough)" :
"") <<
"\n";
367 nVSafe =
MAX2(min, safe);
374 double accel = i.first;
376 if (v >= min && v <= max) {
379 nVSafe =
MIN2(v, nVSafe);
381 nVSafe =
MIN2(v * coopWeight + (1 - coopWeight) * wanted, nVSafe);
384#ifdef DEBUG_PATCHSPEED
386 std::cout <<
SIMTIME <<
" veh=" <<
myVehicle.
getID() <<
" got accel=" << accel <<
" nVSafe=" << nVSafe <<
"\n";
390#ifdef DEBUG_PATCHSPEED
393 std::cout <<
SIMTIME <<
" veh=" <<
myVehicle.
getID() <<
" ignoring low nVSafe=" << v <<
" (accel=" << accel <<
") min=" << min <<
"\n";
397 std::cout <<
SIMTIME <<
" veh=" <<
myVehicle.
getID() <<
" ignoring high nVSafe=" << v <<
" (accel=" << accel <<
") max=" << max <<
"\n";
405#ifdef DEBUG_PATCHSPEED
418#if defined(DEBUG_PATCHSPEED) || defined(DEBUG_STATE)
423 return (max + wanted) / 2.0;
427#if defined(DEBUG_PATCHSPEED) || defined(DEBUG_STATE)
432 return (min + wanted) / 2.0;
435#if defined(DEBUG_PATCHSPEED) || defined(DEBUG_STATE)
440 return (max + wanted) / 2.0;
481#if defined(DEBUG_PATCHSPEED) || defined(DEBUG_STATE)
486 return (max + wanted) / 2.0;
490#if defined(DEBUG_PATCHSPEED) || defined(DEBUG_STATE)
510 if (pinfo->first >= 0) {
519 <<
" informedBy=" << sender->
getID()
520 <<
" info=" << pinfo->second
521 <<
" vSafe=" << pinfo->first
534 assert(cld.first != 0);
543 double remainingSeconds) {
549 plannedSpeed =
MIN2(plannedSpeed, v);
554 std::cout <<
" informLeader speed=" <<
myVehicle.
getSpeed() <<
" planned=" << plannedSpeed <<
"\n";
565 if (
gDebugFlag2) std::cout <<
" blocked by leader nv=" << nv->
getID() <<
" nvSpeed=" << nv->
getSpeed() <<
" needGap="
569 const double dv = plannedSpeed - nv->
getSpeed();
570 const double overtakeDist = (neighLead.second
582 || dv * remainingSeconds < overtakeDist)
583 && (!neighLead.first->isStopped() || (
isOpposite() && neighLead.second >= 0))) {
598 <<
" cannot overtake leader nv=" << nv->
getID()
600 <<
" remainingSeconds=" << remainingSeconds
601 <<
" targetSpeed=" << targetSpeed
602 <<
" nextSpeed=" << nextSpeed
613 <<
" cannot overtake fast leader nv=" << nv->
getID()
615 <<
" remainingSeconds=" << remainingSeconds
616 <<
" targetSpeed=" << targetSpeed
627 <<
" wants to overtake leader nv=" << nv->
getID()
629 <<
" remainingSeconds=" << remainingSeconds
630 <<
" currentGap=" << neighLead.second
632 <<
" overtakeDist=" << overtakeDist
642 }
else if (neighLead.first != 0) {
645 double dv, nextNVSpeed;
665 std::cout <<
" not blocked by leader nv=" << nv->
getID()
667 <<
" gap=" << neighLead.second
668 <<
" nextGap=" << neighLead.second - dv
670 <<
" targetSpeed=" << targetSpeed
674 return MIN2(targetSpeed, plannedSpeed);
686 double remainingSeconds,
687 double plannedSpeed) {
695 if (
gDebugFlag2) std::cout <<
" blocked by follower nv=" << nv->
getID() <<
" nvSpeed=" << nv->
getSpeed() <<
" needGap="
702 if ((neededGap - neighFollow.second) / remainingSeconds < (plannedSpeed - nv->
getSpeed())) {
705 std::cout <<
" wants to cut in before nv=" << nv->
getID() <<
" without any help neededGap=" << neededGap <<
"\n";
724 const double neighNewSpeed1s =
MAX2(0., nv->
getSpeed() - helpDecel);
725 const double dv = plannedSpeed - neighNewSpeed1s;
727 const double decelGap = neighFollow.second + dv;
733 <<
" egoNV=" << plannedSpeed
734 <<
" nvNewSpeed=" << neighNewSpeed
735 <<
" nvNewSpeed1s=" << neighNewSpeed1s
736 <<
" deltaGap=" << dv
737 <<
" decelGap=" << decelGap
738 <<
" secGap=" << secureGap
742 if (decelGap > 0 && decelGap >= secureGap) {
757 std::cout <<
" wants to cut in before nv=" << nv->
getID()
758 <<
" vsafe1=" << vsafe1
759 <<
" vsafe=" << vsafe
764 }
else if (dv > 0 && dv * remainingSeconds > (secureGap - decelGap + POSITION_EPS)) {
769 std::cout <<
" wants to cut in before nv=" << nv->
getID() <<
" (eventually)\n";
777 std::cout <<
" wants to cut in before nv=" << nv->
getID() <<
" (nv cannot overtake right)\n";
793 std::cout <<
" wants right follower to slow down a bit\n";
799 std::cout <<
" wants to cut in before right follower nv=" << nv->
getID() <<
" (eventually)\n";
808 const double overtakeDist = (neighFollow.second
814 const double needDV = overtakeDist / remainingSeconds;
822 <<
" wants to be overtaken by=" << nv->
getID()
823 <<
" overtakeDist=" << overtakeDist
825 <<
" vhelp=" << vhelp
826 <<
" needDV=" << needDV
832 }
else if (neighFollow.first != 0) {
837 std::cout <<
" wants to cut in before non-blocking follower nv=" << neighFollow.first->getID() <<
"\n";
845 const std::vector<CLeaderDist>& blockers,
846 double remainingSeconds) {
858 plannedSpeed =
MIN2(plannedSpeed, safe);
860 for (std::vector<CLeaderDist>::const_iterator it = blockers.begin(); it != blockers.end(); ++it) {
861 plannedSpeed =
MIN2(plannedSpeed,
informLeader(blocked, dir, *it, remainingSeconds));
869 const std::vector<CLeaderDist>& blockers,
870 double remainingSeconds,
871 double plannedSpeed) {
873 for (std::vector<CLeaderDist>::const_iterator it = blockers.begin(); it != blockers.end(); ++it) {
874 informFollower(blocked, dir, *it, remainingSeconds, plannedSpeed);
899 const double halfWidth =
getWidth() * 0.5;
912 std::vector<double> newExpectedSpeeds;
921 const std::vector<MSLane*>& lanes = currEdge->
getLanes();
922 for (std::vector<MSLane*>::const_iterator it_lane = lanes.begin(); it_lane != lanes.end(); ++it_lane) {
924 for (
int i = 0; i < subLanes; ++i) {
925 newExpectedSpeeds.push_back((*it_lane)->getVehicleMaxSpeed(&
myVehicle));
931 for (
int i = 0; i < subLanes; ++i) {
932 newExpectedSpeeds.push_back(lanes.back()->getVehicleMaxSpeed(&
myVehicle));
940 if (subLaneShift < std::numeric_limits<int>::max()) {
942 const int newI = i + subLaneShift;
943 if (newI > 0 && newI < (
int)newExpectedSpeeds.size()) {
979 for (
const MSLink*
const link : lane->getLinkCont()) {
980 if (&link->getLane()->getEdge() == curEdge) {
982 const MSLane* target = link->getLane();
983 const std::vector<MSLane*>& lanes2 = curEdge->
getLanes();
984 for (std::vector<MSLane*>::const_iterator it_lane2 = lanes2.begin(); it_lane2 != lanes2.end(); ++it_lane2) {
985 const MSLane* lane2 = *it_lane2;
986 if (lane2 == target) {
987 return prevShift + curShift;
998 return std::numeric_limits<int>::max();
1033#if defined(DEBUG_MANEUVER) || defined(DEBUG_STATE)
1066 const std::vector<MSVehicle::LaneQ>& preb,
1069 double& latDist,
double& maneuverDist,
int& blocked) {
1071 if (laneOffset != 0) {
1073 const double halfWidth =
getWidth() * 0.5;
1075 if (laneOffset < 0) {
1085 int bestLaneOffset = 0;
1086 double currentDist = 0;
1087 double neighDist = 0;
1099 const int prebOffset = (checkOpposite ? 0 : laneOffset);
1100 for (
int p = 0; p < (int) preb.size(); ++p) {
1101 if (preb[p].lane == prebLane && p + laneOffset >= 0) {
1102 assert(p + prebOffset < (
int)preb.size());
1104 neigh = preb[p + prebOffset];
1105 currentDist = curr.
length;
1106 neighDist = neigh.
length;
1109 if (bestLaneOffset == 0 && preb[p + prebOffset].bestLaneOffset == 0 && !checkOpposite) {
1110#ifdef DEBUG_WANTSCHANGE
1114 <<
" bestLaneOffsetOld=" << bestLaneOffset
1115 <<
" bestLaneOffsetNew=" << laneOffset
1119 bestLaneOffset = prebOffset;
1121 best = preb[p + bestLaneOffset];
1125 assert(curr.
lane !=
nullptr);
1126 assert(neigh.
lane !=
nullptr);
1127 assert(best.
lane !=
nullptr);
1128 double driveToNextStop = -std::numeric_limits<double>::max();
1137#ifdef DEBUG_WANTS_CHANGE
1142 <<
" stopPos=" << stopPos
1143 <<
" currentDist=" << currentDist
1144 <<
" neighDist=" << neighDist
1148 currentDist =
MAX2(currentDist, stopPos);
1149 neighDist =
MAX2(neighDist, stopPos);
1152 const bool right = (laneOffset == -1);
1153 const bool left = (laneOffset == 1);
1156 const bool changeToBest = (right && bestLaneOffset < 0) || (left && bestLaneOffset > 0) || (laneOffset == 0 && bestLaneOffset == 0);
1182#ifdef DEBUG_WANTSCHANGE
1189 <<
"\n leaders=" << leaders.
toString()
1190 <<
"\n followers=" << followers.
toString()
1191 <<
"\n blockers=" << blockers.
toString()
1192 <<
"\n neighLeaders=" << neighLeaders.
toString()
1193 <<
"\n neighFollowers=" << neighFollowers.
toString()
1194 <<
"\n neighBlockers=" << neighBlockers.
toString()
1195 <<
"\n changeToBest=" << changeToBest
1196 <<
" latLaneDist=" << latLaneDist
1205 if (lastBlocked != firstBlocked) {
1256 for (
int i = 0; i < neighLeaders.
numSublanes(); ++i) {
1258 if (vehDist.first !=
nullptr && vehDist.first->isStopped()) {
1276 currentDist += roundaboutBonus;
1277 neighDist += roundaboutBonus;
1296 if ((ret &
LCA_STAY) != 0 && latDist == 0) {
1309 if (changeToBest && abs(bestLaneOffset) > 1
1315#ifdef DEBUG_WANTSCHANGE
1317 std::cout <<
" reserving space for unseen blockers myLeadingBlockerLength=" <<
myLeadingBlockerLength <<
"\n";
1326#ifdef DEBUG_WANTSCHANGE
1332 if (*firstBlocked != neighLeadLongest &&
tieBrakeLeader(*firstBlocked)) {
1341 std::vector<CLeaderDist> collectLeadBlockers;
1342 std::vector<CLeaderDist> collectFollowBlockers;
1343 int blockedFully = 0;
1344 maneuverDist = latDist;
1346 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
1347 leaders, followers, blockers,
1348 neighLeaders, neighFollowers, neighBlockers, &collectLeadBlockers, &collectFollowBlockers,
1349 false, gapFactor, &blockedFully);
1351 const double absLaneOffset = fabs(bestLaneOffset != 0 ? bestLaneOffset : latDist /
SUMO_const_laneWidth);
1352 const double remainingSeconds = ((ret &
LCA_TRACI) == 0 ?
1355 const double plannedSpeed =
informLeaders(blocked, myLca, collectLeadBlockers, remainingSeconds);
1357 if (plannedSpeed >= 0) {
1359 informFollowers(blocked, myLca, collectFollowBlockers, remainingSeconds, plannedSpeed);
1361 if (plannedSpeed > 0) {
1362 commitManoeuvre(blocked, blockedFully, leaders, neighLeaders, neighLane, maneuverDist);
1364#if defined(DEBUG_WANTSCHANGE) || defined(DEBUG_STATE)
1371 <<
" remainingSeconds=" << remainingSeconds
1372 <<
" plannedSpeed=" << plannedSpeed
1383 if (roundaboutBonus > 0) {
1385#ifdef DEBUG_WANTS_CHANGE
1389 <<
" roundaboutBonus=" << roundaboutBonus
1400 latDist = latLaneDist;
1401 maneuverDist = latLaneDist;
1402 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
1403 leaders, followers, blockers,
1404 neighLeaders, neighFollowers, neighBlockers);
1408 ret &= ~LCA_COOPERATIVE;
1424 const double inconvenience = (latLaneDist < 0
1427#ifdef DEBUG_COOPERATE
1435 <<
" inconvenience=" << inconvenience
1437 <<
" wantsChangeToHelp=" << (right ?
"right" :
"left")
1454 && (changeToBest ||
currentDistAllows(neighDist, abs(bestLaneOffset) + 1, laDist))) {
1457#ifdef DEBUG_COOPERATE
1459 std::cout <<
" wants cooperative change\n";
1466 maneuverDist = latDist;
1467 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
1468 leaders, followers, blockers,
1469 neighLeaders, neighFollowers, neighBlockers);
1495 const double vehWidth =
getWidth();
1497 const double leftVehSide = rightVehSide + vehWidth;
1499 double defaultNextSpeed = std::numeric_limits<double>::max();
1501 int leftmostOnEdge = (int)sublaneSides.size() - 1;
1502 while (leftmostOnEdge > 0 && sublaneSides[leftmostOnEdge] > leftVehSide) {
1505 int rightmostOnEdge = leftmostOnEdge;
1506 while (rightmostOnEdge > 0 && sublaneSides[rightmostOnEdge] > rightVehSide + NUMERICAL_EPS) {
1508#ifdef DEBUG_WANTSCHANGE
1510 std::cout <<
" adapted to current sublane=" << rightmostOnEdge <<
" defaultNextSpeed=" << defaultNextSpeed <<
"\n";
1511 std::cout <<
" sublaneSides[rightmostOnEdge]=" << sublaneSides[rightmostOnEdge] <<
" rightVehSide=" << rightVehSide <<
"\n";
1517#ifdef DEBUG_WANTSCHANGE
1519 std::cout <<
" adapted to current sublane=" << rightmostOnEdge <<
" defaultNextSpeed=" << defaultNextSpeed <<
"\n";
1520 std::cout <<
" sublaneSides[rightmostOnEdge]=" << sublaneSides[rightmostOnEdge] <<
" rightVehSide=" << rightVehSide <<
"\n";
1523 double maxGain = -std::numeric_limits<double>::max();
1524 double maxGainRight = -std::numeric_limits<double>::max();
1525 double maxGainLeft = -std::numeric_limits<double>::max();
1526 double latDistNice = std::numeric_limits<double>::max();
1529 double leftMax =
MAX2(
1536 assert(leftMax <= edge.
getWidth());
1538 int sublaneCompact =
MAX2(iMin, rightmostOnEdge - 1);
1544 const double maxLatDist = leftMax - leftVehSide;
1545 const double minLatDist = rightMin - rightVehSide;
1546 const int iStart = laneOffset == 0 ? iMin : 0;
1547 const double rightEnd = laneOffset == 0 ? leftMax : (checkOpposite ?
getLeftBorder() : edge.
getWidth());
1548#ifdef DEBUG_WANTSCHANGE
1550 <<
" checking sublanes rightmostOnEdge=" << rightmostOnEdge
1551 <<
" rightEnd=" << rightEnd
1552 <<
" leftmostOnEdge=" << leftmostOnEdge
1553 <<
" iStart=" << iStart
1555 <<
" sublaneSides=" << sublaneSides.size()
1556 <<
" leftMax=" << leftMax
1557 <<
" minLatDist=" << minLatDist
1558 <<
" maxLatDist=" << maxLatDist
1559 <<
" sublaneCompact=" << sublaneCompact
1562 for (
int i = iStart; i < (int)sublaneSides.size(); ++i) {
1563 if (sublaneSides[i] + vehWidth < rightEnd) {
1569 while (vMin > 0 && j < (
int)sublaneSides.size() && sublaneSides[j] < sublaneSides[i] + vehWidth) {
1571#ifdef DEBUG_WANTSCHANGE
1579 if (laneOffset != 0 &&
overlap(sublaneSides[i], sublaneSides[i] + vehWidth, laneBoundary, laneBoundary)) {
1583 double currentLatDist = sublaneSides[i] - rightVehSide;
1584 if ((laneOffset == 0 && (currentLatDist > maxLatDist || currentLatDist < minLatDist))
1585 || (laneOffset < 0 && currentLatDist > maxLatDist)
1586 || (laneOffset > 0 && currentLatDist < minLatDist)) {
1587#ifdef DEBUG_WANTSCHANGE
1589 std::cout <<
" i=" << i <<
" currentLatDist=" << currentLatDist <<
" outOfBounds\n";
1594 currentLatDist =
MIN2(
MAX2(currentLatDist, minLatDist), maxLatDist);
1598 relativeGain *= 0.5;
1601 if (relativeGain > maxGain && currentLatDist * laneOffset >= 0) {
1602 maxGain = relativeGain;
1605 latDist = currentLatDist;
1606#ifdef DEBUG_WANTSCHANGE
1608 std::cout <<
" i=" << i <<
" vMin=" << vMin <<
" newLatDist=" << latDist <<
" relGain=" << relativeGain <<
"\n";
1614 if (currentLatDist > 0
1618 && maxGain - relativeGain < NUMERICAL_EPS) {
1619 latDist = currentLatDist;
1622#ifdef DEBUG_WANTSCHANGE
1624 std::cout <<
" i=" << i <<
" rightmostOnEdge=" << rightmostOnEdge <<
" vMin=" << vMin <<
" relGain=" << relativeGain <<
" sublaneCompact=" << sublaneCompact <<
" curLatDist=" << currentLatDist <<
"\n";
1628 maxGainRight =
MAX2(maxGainRight, relativeGain);
1630 maxGainLeft =
MAX2(maxGainLeft, relativeGain);
1632 const double subAlignDist = sublaneSides[i] - rightVehSide;
1633 if (fabs(subAlignDist) < fabs(latDistNice)) {
1634 latDistNice = subAlignDist;
1635#ifdef DEBUG_WANTSCHANGE
1637 <<
" nicest sublane=" << i
1638 <<
" side=" << sublaneSides[i]
1639 <<
" rightSide=" << rightVehSide
1640 <<
" latDistNice=" << latDistNice
1641 <<
" maxGainR=" << (maxGainRight == -std::numeric_limits<double>::max() ?
"n/a" :
toString(maxGainRight))
1642 <<
" maxGainL=" << (maxGainLeft == -std::numeric_limits<double>::max() ?
"n/a" :
toString(maxGainLeft))
1649 if (maxGainRight != -std::numeric_limits<double>::max()) {
1650#ifdef DEBUG_WANTSCHANGE
1656#ifdef DEBUG_WANTSCHANGE
1662 if (maxGainLeft != -std::numeric_limits<double>::max()) {
1663#ifdef DEBUG_WANTSCHANGE
1669#ifdef DEBUG_WANTSCHANGE
1676 if ((fabs(maxGainRight) < NUMERICAL_EPS || maxGainRight == -std::numeric_limits<double>::max())
1677 && (right || (alternatives &
LCA_RIGHT) == 0)) {
1680 if ((fabs(maxGainLeft) < NUMERICAL_EPS || maxGainLeft == -std::numeric_limits<double>::max())
1681 && (left || (alternatives &
LCA_LEFT) == 0)) {
1686#ifdef DEBUG_WANTSCHANGE
1689 <<
" defaultNextSpeed=" << defaultNextSpeed
1690 <<
" maxGain=" << maxGain
1691 <<
" maxGainRight=" << maxGainRight
1692 <<
" maxGainLeft=" << maxGainLeft
1695 <<
" latDist=" << latDist
1696 <<
" latDistNice=" << latDistNice
1697 <<
" sublaneCompact=" << sublaneCompact
1710 double acceptanceTime;
1719 double minFactor = 1.0;
1720 for (
int i = 0; i < followers.
numSublanes(); ++i) {
1722 if (follower.first !=
nullptr && follower.second < 2 * follower.first->getCarFollowModel().brakeGap(follower.first->getSpeed())) {
1725 const double fRSF = follower.first->getLane()->getVehicleMaxSpeed(follower.first) / follower.first->getLane()->getSpeedLimit();
1726 if (fRSF > roadSpeedFactor) {
1729 if (factor < minFactor) {
1735 acceptanceTime *= minFactor;
1739 double fullSpeedDrivingSeconds =
MIN2(acceptanceTime, fullSpeedGap / vMax);
1741 if (neighLead.first != 0 && neighLead.first->getSpeed() < vMax) {
1742 fullSpeedGap =
MAX2(0.,
MIN2(fullSpeedGap,
1744 vMax, neighLead.first->
getSpeed(), neighLead.first->getCarFollowModel().getMaxDecel())));
1745 fullSpeedDrivingSeconds =
MIN2(fullSpeedDrivingSeconds, fullSpeedGap / (vMax - neighLead.first->getSpeed()));
1754#ifdef DEBUG_WANTSCHANGE
1757 <<
" considering keepRight:"
1759 <<
" neighDist=" << neighDist
1761 <<
" leaderSpeed=" << (neighLead.first == 0 ? -1 : neighLead.first->getSpeed())
1763 myVehicle.
getSpeed(), neighLead.first->getSpeed(), neighLead.first->getCarFollowModel().getMaxDecel()))
1764 <<
" acceptanceTime=" << acceptanceTime
1765 <<
" fullSpeedGap=" << fullSpeedGap
1766 <<
" fullSpeedDrivingSeconds=" << fullSpeedDrivingSeconds
1767 <<
" dProb=" << deltaProb
1768 <<
" isSlide=" << isSlide
1779 latDist = latLaneDist;
1780 maneuverDist = latLaneDist;
1781 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
1782 leaders, followers, blockers,
1783 neighLeaders, neighFollowers, neighBlockers);
1786 ret &= ~LCA_KEEPRIGHT;
1792#ifdef DEBUG_WANTSCHANGE
1797 <<
" neighDist=" << neighDist
1801 <<
" latDist=" << latDist
1811 int blockedFully = 0;
1812 maneuverDist = latDist;
1813 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
1814 leaders, followers, blockers,
1815 neighLeaders, neighFollowers, neighBlockers,
1816 nullptr,
nullptr,
false, 0, &blockedFully);
1822 ret &= ~LCA_SPEEDGAIN;
1829#ifdef DEBUG_WANTSCHANGE
1834 <<
" latDist=" << latDist
1835 <<
" neighDist=" << neighDist
1838 <<
" stayInLane=" << stayInLane
1849 int blockedFully = 0;
1850 maneuverDist = latDist;
1851 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
1852 leaders, followers, blockers,
1853 neighLeaders, neighFollowers, neighBlockers,
1854 nullptr,
nullptr,
false, 0, &blockedFully);
1859 ret &= ~LCA_SPEEDGAIN;
1864 double latDistSublane = 0.;
1866 const double halfVehWidth =
getWidth() * 0.5;
1869 && bestLaneOffset == 0
1889#ifdef DEBUG_WANTSCHANGE
1900 latDistSublane = -halfLaneWidth + halfVehWidth -
getPosLat();
1903 latDistSublane = halfLaneWidth - halfVehWidth -
getPosLat();
1910 latDistSublane = latDistNice;
1913 latDistSublane = sublaneSides[sublaneCompact] - rightVehSide;
1919 if (fabs(posLat) > hLW) {
1922 latDistSublane -= (posLat - hLW);
1924 latDistSublane += (-posLat - hLW);
1929 if (rightVehSide < 0) {
1930 latDistSublane -= rightVehSide;
1931 }
else if (leftVehSide > edgeWidth) {
1932 latDistSublane -= leftVehSide - edgeWidth;
1951 latDistSublane * latDist > 0) {
1953#if defined(DEBUG_WANTSCHANGE) || defined(DEBUG_STATE) || defined(DEBUG_MANEUVER)
1958 <<
" latDist=" << latDist
1959 <<
" latDistSublane=" << latDistSublane
1960 <<
" relGainSublane=" <<
computeSpeedGain(latDistSublane, defaultNextSpeed)
1961 <<
" maneuverDist=" << maneuverDist
1973#if defined(DEBUG_WANTSCHANGE)
1975 <<
" speedGain=" <<
computeSpeedGain(latDistSublane, defaultNextSpeed) <<
")\n";
1983#if defined(DEBUG_WANTSCHANGE)
1985 std::cout <<
" aborting sublane change due to prior maneuver\n";
1990 latDist = latDistSublane * (
isOpposite() ? -1 : 1);
1995#ifdef DEBUG_WANTSCHANGE
1998 <<
" latDist=" << latDist
2006#ifdef DEBUG_WANTSCHANGE
2012 maneuverDist = latDist;
2013 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset,
2014 leaders, followers, blockers,
2015 neighLeaders, neighFollowers, neighBlockers);
2018 ret &= ~LCA_SUBLANE;
2044#ifdef DEBUG_WANTSCHANGE
2061 if ((*blocked) !=
nullptr) {
2063#ifdef DEBUG_SLOWDOWN
2072 if (gap > POSITION_EPS) {
2086 (*blocked)->getSpeed(), (*blocked)->getCarFollowModel().getMaxDecel());
2089 std::cout <<
" slowing down for blocked " <<
Named::getIDSecure(*blocked) <<
" targetSpeed=" << targetSpeed <<
"\n";
2110 if (cand !=
nullptr && cand->getBidiLane() == lane) {
2123 const MSLink* link = next !=
nullptr ? lane->
getLinkTo(next) :
nullptr;
2127 assert(preb.size() == lanes.size() ||
isOpposite());
2128#ifdef DEBUG_EXPECTED_SLSPEED
2131 <<
" sublaneOffset=" << sublaneOffset <<
" laneIndex=" << laneIndex <<
" lane=" << lane->
getID() <<
" ahead=" << ahead.
toString() <<
"\n";
2135 for (
int sublane = 0; sublane < (int)ahead.
numSublanes(); ++sublane) {
2136 const int edgeSublane = sublane + sublaneOffset;
2144#ifdef DEBUG_EXPECTED_SLSPEED
2146 std::cout <<
" updateExpectedSublaneSpeeds sublane=" << sublane <<
" doesNotContinue\n";
2153 const MSVehicle* leader = ahead[sublane].first;
2154 const double gap = ahead[sublane].second;
2156 if (leader ==
nullptr) {
2161 const int prebIndex =
isOpposite() ? (int)preb.size() - 1 : laneIndex;
2172#ifdef DEBUG_EXPECTED_SLSPEED
2174 std::cout <<
SIMTIME <<
" updateExpectedSublaneSpeeds sublane=" << sublane <<
" leader=" << leader->
getID() <<
" bidi=" << bidi->
getID() <<
" gap=" << gap <<
" vSafe=" << vSafe <<
"\n";
2184#ifdef DEBUG_EXPECTED_SLSPEED
2186 std::cout <<
" updateExpectedSublaneSpeeds edgeSublane=" << edgeSublane <<
" leader=" << leader->
getID() <<
" gap=" << gap <<
" vSafe=" << vSafe <<
"\n";
2195 double foeRight, foeLeft;
2199 if (pedLeader.first != 0) {
2204#ifdef DEBUG_EXPECTED_SLSPEED
2206 std::cout <<
" updateExpectedSublaneSpeeds edgeSublane=" << edgeSublane <<
" pedLeader=" << pedLeader.first->getID() <<
" gap=" << pedGap <<
" vSafe=" << vSafe <<
"\n";
2214 double foeRight, foeLeft;
2216 const double foeRightBidi = bidi->
getWidth() - foeLeft;
2217 const double foeLeftBidi = bidi->
getWidth() - foeRight;
2222 if (pedLeader.first != 0) {
2227#ifdef DEBUG_EXPECTED_SLSPEED
2229 std::cout <<
" updateExpectedSublaneSpeeds edgeSublane=" << edgeSublane <<
" pedLeader=" << pedLeader.first->getID() <<
" (bidi) gap=" << pedGap <<
" vSafe=" << vSafe <<
"\n";
2234 vSafe =
MIN2(vMax, vSafe);
2241#ifdef DEBUG_EXPECTED_SLSPEED
2243 std::cout <<
" updateExpectedSublaneSpeeds edgeSublane=" << edgeSublane <<
" lane " << lane->
getID() <<
" forbidden\n";
2255 return ((side < -NUMERICAL_EPS
2264 const double deltaV = vMax - vLeader;
2265 if (deltaV > 0 && gap / deltaV < mySpeedGainLookahead && mySpeedGainLookahead > 0) {
2269 const double gapClosingTime =
MAX2(0.0, gap / deltaV);
2270 const double vSafe2 = (gapClosingTime * vSafe + (foreCastTime - gapClosingTime) * vLeader) / foreCastTime;
2271#ifdef DEBUG_EXPECTED_SLSPEED
2273 std::cout <<
" foreCastTime=" << foreCastTime <<
" gapClosingTime=" << gapClosingTime <<
" extrapolated vSafe=" << vSafe2 <<
"\n";
2284 double result = std::numeric_limits<double>::max();
2286 const double vehWidth =
getWidth();
2288 const double leftVehSide = rightVehSide + vehWidth;
2289 for (
int i = 0; i < (int)sublaneSides.size(); ++i) {
2291 if (
overlap(rightVehSide, leftVehSide, sublaneSides[i], leftSide)) {
2296 return result - defaultNextSpeed;
2303 double maxLength = -1;
2314 return iMax >= 0 ? ldi[iMax] : std::make_pair(
nullptr, -1);
2331 double minSpeed = std::numeric_limits<double>::max();
2333 if (ldi[i].first != 0) {
2334 const double speed = ldi[i].first->getSpeed();
2335 if (speed < minSpeed) {
2349 if (v !=
nullptr && v->isStopped()) {
2366 std::vector<CLeaderDist>* collectLeadBlockers,
2367 std::vector<CLeaderDist>* collectFollowBlockers,
2368 bool keepLatGapManeuver,
2370 int* retBlockedFully) {
2373 latDist =
MAX2(
MIN2(latDist, maxDist), -maxDist);
2382 if (laneOffset != 0) {
2393 if (laneOffset != 0) {
2397#ifdef DEBUG_BLOCKING
2409 }
else if (!forcedTraCIChange) {
2415 }
else if (!forcedTraCIChange) {
2421#ifdef DEBUG_BLOCKING
2423 std::cout <<
" checkBlocking latDist=" << latDist <<
" maneuverDist=" << maneuverDist <<
"\n";
2437 if (laneOffset != 0) {
2444 int blockedFully = 0;
2449 if (laneOffset != 0) {
2455 if (retBlockedFully !=
nullptr) {
2456 *retBlockedFully = blockedFully;
2458#ifdef DEBUG_BLOCKING
2461 <<
" canChangeFully=" <<
myCanChangeFully <<
" keepLatGapManeuver=" << keepLatGapManeuver <<
"\n";
2469 blocked |= blockedFully;
2474#ifdef DEBUG_BLOCKING
2479 if (collectFollowBlockers !=
nullptr && collectLeadBlockers !=
nullptr) {
2481 for (std::vector<CLeaderDist>::const_iterator it2 = collectLeadBlockers->begin(); it2 != collectLeadBlockers->end(); ++it2) {
2482 for (std::vector<CLeaderDist>::iterator it = collectFollowBlockers->begin(); it != collectFollowBlockers->end();) {
2483 if ((*it2).first == (*it).first) {
2484#ifdef DEBUG_BLOCKING
2486 std::cout <<
" removed follower " << (*it).first->getID() <<
" because it is already a leader\n";
2489 it = collectFollowBlockers->erase(it);
2503 int laneOffset,
double latDist,
double foeOffset,
bool leaders,
2504 double& safeLatGapRight,
double& safeLatGapLeft,
2505 std::vector<CLeaderDist>* collectBlockers)
const {
2512 const double vehWidth =
getWidth();
2514 const double leftVehSide = rightVehSide + vehWidth;
2515 const double rightVehSideDest = rightVehSide + latDist;
2516 const double leftVehSideDest = leftVehSide + latDist;
2517 const double rightNoOverlap =
MIN2(rightVehSideDest, rightVehSide);
2518 const double leftNoOverlap =
MAX2(leftVehSideDest, leftVehSide);
2519#ifdef DEBUG_BLOCKING
2521 std::cout <<
" checkBlockingVehicles"
2522 <<
" laneOffset=" << laneOffset
2523 <<
" latDist=" << latDist
2524 <<
" foeOffset=" << foeOffset
2525 <<
" vehRight=" << rightVehSide
2526 <<
" vehLeft=" << leftVehSide
2527 <<
" rightNoOverlap=" << rightNoOverlap
2528 <<
" leftNoOverlap=" << leftNoOverlap
2529 <<
" destRight=" << rightVehSideDest
2530 <<
" destLeft=" << leftVehSideDest
2531 <<
" leaders=" << leaders
2537 for (
int i = 0; i < vehicles.
numSublanes(); ++i) {
2539 if (vehDist.first != 0 &&
myCFRelated.count(vehDist.first) == 0) {
2540 const MSVehicle* leader = vehDist.first;
2546 double foeRight, foeLeft;
2548 const bool overlapBefore =
overlap(rightVehSide, leftVehSide, foeRight, foeLeft);
2549 const bool overlapDest =
overlap(rightVehSideDest, leftVehSideDest, foeRight, foeLeft);
2550 const bool overlapAny =
overlap(rightNoOverlap, leftNoOverlap, foeRight, foeLeft);
2551#ifdef DEBUG_BLOCKING
2553 std::cout <<
" foe=" << vehDist.first->getID()
2554 <<
" gap=" << vehDist.second
2556 <<
" foeRight=" << foeRight
2557 <<
" foeLeft=" << foeLeft
2558 <<
" overlapBefore=" << overlapBefore
2559 <<
" overlap=" << overlapAny
2560 <<
" overlapDest=" << overlapDest
2565 if (vehDist.second < 0) {
2566 if (overlapBefore && !overlapDest && !
outsideEdge()) {
2567#ifdef DEBUG_BLOCKING
2569 std::cout <<
" ignoring current overlap to come clear\n";
2573#ifdef DEBUG_BLOCKING
2579 if (collectBlockers ==
nullptr) {
2582 collectBlockers->push_back(vehDist);
2598 const double expectedGap =
MSCFModel::gapExtrapolation(timeTillAction, vehDist.second, leader->
getSpeed(), follower->
getSpeed(), leaderAccel, followerAccel, std::numeric_limits<double>::max(), std::numeric_limits<double>::max());
2601 const double followerExpectedSpeed = follower->
getSpeed() + timeTillAction * followerAccel;
2602 const double leaderExpectedSpeed =
MAX2(0., leader->
getSpeed() + timeTillAction * leaderAccel);
2605#if defined(DEBUG_ACTIONSTEPS) && defined(DEBUG_BLOCKING)
2607 std::cout <<
" timeTillAction=" << timeTillAction
2608 <<
" followerAccel=" << followerAccel
2609 <<
" followerExpectedSpeed=" << followerExpectedSpeed
2610 <<
" leaderAccel=" << leaderAccel
2611 <<
" leaderExpectedSpeed=" << leaderExpectedSpeed
2612 <<
"\n gap=" << vehDist.second
2613 <<
" gapChange=" << (expectedGap - vehDist.second)
2614 <<
" expectedGap=" << expectedGap
2615 <<
" expectedSecureGap=" << expectedSecureGap
2616 <<
" safeLatGapLeft=" << safeLatGapLeft
2617 <<
" safeLatGapRight=" << safeLatGapRight
2624 if (expectedGap < secureGap2) {
2626 if (foeRight > leftVehSide) {
2627 safeLatGapLeft =
MIN2(safeLatGapLeft, foeRight - leftVehSide);
2628 }
else if (foeLeft < rightVehSide) {
2629 safeLatGapRight =
MIN2(safeLatGapRight, rightVehSide - foeLeft);
2632#ifdef DEBUG_BLOCKING
2634 std::cout <<
" blocked by " << vehDist.first->getID() <<
" gap=" << vehDist.second <<
" expectedGap=" << expectedGap
2635 <<
" expectedSecureGap=" << expectedSecureGap <<
" secGap2=" << secureGap2 <<
" safetyFactor=" <<
getSafetyFactor()
2636 <<
" safeLatGapLeft=" << safeLatGapLeft <<
" safeLatGapRight=" << safeLatGapRight
2640 result |= blockType;
2641 if (collectBlockers ==
nullptr) {
2644#ifdef DEBUG_BLOCKING
2645 }
else if (
gDebugFlag2 && expectedGap < expectedSecureGap) {
2646 std::cout <<
" ignore blocker " << vehDist.first->getID() <<
" gap=" << vehDist.second <<
" expectedGap=" << expectedGap
2647 <<
" expectedSecureGap=" << expectedSecureGap <<
" secGap2=" << secureGap2 <<
" safetyFactor=" <<
getSafetyFactor() <<
"\n";
2650 if (collectBlockers !=
nullptr) {
2653 collectBlockers->push_back(vehDist);
2670 const double leftVehSide = rightVehSide + vehWidth;
2671#ifdef DEBUG_BLOCKING
2673 std::cout <<
" updateCFRelated foeOffset=" << foeOffset <<
" vehicles=" << vehicles.
toString() <<
"\n";
2676 for (
int i = 0; i < vehicles.
numSublanes(); ++i) {
2678 if (vehDist.first != 0 && (
myCFRelated.count(vehDist.first) == 0 || vehDist.second < 0)) {
2679 double foeRight, foeLeft;
2681#ifdef DEBUG_BLOCKING
2683 std::cout <<
" foe=" << vehDist.first->getID() <<
" gap=" << vehDist.second
2685 <<
" foeOffset=" << foeOffset
2686 <<
" egoR=" << rightVehSide <<
" egoL=" << leftVehSide
2687 <<
" iR=" << foeRight <<
" iL=" << foeLeft
2693 if (
overlap(rightVehSide, leftVehSide, foeRight, foeLeft) && !
outsideEdge() && (vehDist.second >= 0
2699 && -vehDist.second < vehDist.first->getVehicleType().getMinGap()
2702#ifdef DEBUG_BLOCKING
2704 std::cout <<
" ignoring cfrelated foe=" << vehDist.first->getID() <<
"\n";
2709 const int erased = (int)
myCFRelated.erase(vehDist.first);
2710#ifdef DEBUG_BLOCKING
2712 std::cout <<
" restoring cfrelated foe=" << vehDist.first->getID() <<
"\n";
2725 assert(right <= left);
2726 assert(right2 <= left2);
2727 return left2 >= right + NUMERICAL_EPS && left >= right2 + NUMERICAL_EPS;
2748 return changeReason;
2755 if (sd1.
state == 0) {
2757 }
else if (sd2.
state == 0) {
2767#ifdef DEBUG_DECISION
2773 <<
" dir1=" << sd1.
dir
2777 <<
" dir2=" << sd2.
dir
2793 if (reason1 < reason2) {
2795 return (!can1 && can2 && sd1.
sameDirection(sd2)) ? sd2 : sd1;
2797 }
else if (reason1 > reason2) {
2799 return (!can2 && can1 && sd1.
sameDirection(sd2)) ? sd1 : sd2;
2807 }
else if (sd2.
dir == 0) {
2812 assert(sd1.
dir == -1);
2813 assert(sd2.
dir == 1);
2816 }
else if (sd2.
latDist >= 0) {
2872 double& currentDist,
2875 double roundaboutBonus,
2880 const bool right = (laneOffset == -1);
2881 const bool left = (laneOffset == 1);
2884 if (laneOffset != 0) {
2890 const double neighLeftPlace =
MAX2(0., neighDist - forwardPos - maxJam);
2894#ifdef DEBUG_STRATEGIC_CHANGE
2898 <<
" forwardPos=" << forwardPos
2900 <<
" laDist=" << laDist
2901 <<
" currentDist=" << currentDist
2902 <<
" usableDist=" << usableDist
2903 <<
" bestLaneOffset=" << bestLaneOffset
2904 <<
" best.length=" << best.
length
2905 <<
" maxJam=" << maxJam
2906 <<
" neighLeftPlace=" << neighLeftPlace
2913 if (laneOffset == 0) {
2920#ifdef DEBUG_STRATEGIC_CHANGE
2922 std::cout <<
SIMTIME <<
" returnToLaneBounds\n";
2931 if (link !=
nullptr &&
getWidth() < next->
getWidth() && distOnLane < 100) {
2937 latDist = -rightVehSide;
2940#ifdef DEBUG_STRATEGIC_CHANGE
2947 latDist = -(leftVehSide - next->
getWidth());
2950#ifdef DEBUG_STRATEGIC_CHANGE
2959 }
else if (laneOffset != 0 && changeToBest && bestLaneOffset == curr.
bestLaneOffset
2962 if (!
mustOvertakeStopped(
false, neighLane, neighLeaders, leaders, forwardPos, neighDist, right, latLaneDist, currentDist, latDist)) {
2963 latDist = latLaneDist;
2965#ifdef DEBUG_STRATEGIC_CHANGE
2967 std::cout <<
SIMTIME <<
" mustChangeToBest\n";
2971#ifdef DEBUG_STRATEGIC_CHANGE
2973 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" avoidStoppedNeigh\n";
2993#ifdef DEBUG_STRATEGIC_CHANGE
2996 <<
" avoid overtaking on the right nv=" << nv->
getID()
3008 if (laneOffset != 0 &&
myStrategicParam >= 0 && noOpposites &&
mustOvertakeStopped(
true, neighLane, leaders, neighLeaders, forwardPos, neighDist, right, latLaneDist, currentDist, latDist)) {
3009#ifdef DEBUG_STRATEGIC_CHANGE
3011 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" mustOvertakeStopped\n";
3020 }
else if (!changeToBest && (
currentDistDisallows(neighLeftPlace, abs(bestLaneOffset) + 2, laDist))) {
3027#ifdef DEBUG_STRATEGIC_CHANGE
3029 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" could not change back and forth in time (1) neighLeftPlace=" << neighLeftPlace <<
"\n";
3035 && bestLaneOffset == 0
3038 && roundaboutBonus == 0
3045#ifdef DEBUG_STRATEGIC_CHANGE
3047 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" does not want to leave the bestLane (neighDist=" << neighDist <<
")\n";
3052 && bestLaneOffset == 0
3058#ifdef DEBUG_STRATEGIC_CHANGE
3060 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" does not want to get stranded on the on-ramp of a highway\n";
3074 MSLane* shadowPrev =
nullptr;
3076 if (*it ==
nullptr) {
3080 if (shadow ==
nullptr || currentShadowDist >= requiredDist) {
3083 if (shadowPrev !=
nullptr) {
3086 currentShadowDist += shadow->
getLength();
3087 shadowPrev = shadow;
3088#ifdef DEBUG_STRATEGIC_CHANGE
3090 std::cout <<
" shadow=" << shadow->
getID() <<
" currentShadowDist=" << currentShadowDist <<
"\n";
3094#ifdef DEBUG_STRATEGIC_CHANGE
3096 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" currentShadowDist=" << currentShadowDist <<
" requiredDist=" << requiredDist <<
" overlap=" <<
overlap <<
"\n";
3099 if (currentShadowDist < requiredDist && currentShadowDist < usableDist) {
3102#ifdef DEBUG_STRATEGIC_CHANGE
3104 std::cout <<
" must change for shadowLane end latDist=" << latDist <<
" myLeftSpace=" <<
myLeftSpace <<
"\n";
3112#if defined(DEBUG_STRATEGIC_CHANGE) || defined(DEBUG_TRACI)
3124 }
else if (((retTraCI &
LCA_RIGHT) != 0 && laneOffset < 0)
3125 || ((retTraCI &
LCA_LEFT) != 0 && laneOffset > 0)) {
3127 latDist = latLaneDist;
3130#if defined(DEBUG_STRATEGIC_CHANGE) || defined(DEBUG_TRACI)
3141 double posOnLane,
double neighDist,
bool right,
double latLaneDist,
double& currentDist,
double& latDist) {
3142 bool mustOvertake =
false;
3147 const bool curHasStopped = stoppedLeader !=
nullptr;
3148 const int dir = latLaneDist < 0 ? -1 : 1;
3152 if (curHasStopped) {
3154 for (
int i = rightmost; i <= leftmost; i++) {
3158 const double remaining =
MIN2(neighDist, currentDist) - posOnLane;
3159#ifdef DEBUG_STRATEGIC_CHANGE
3161 std::cout <<
" overtakeDist=" << overtakeDist <<
" remaining=" << remaining
3163 <<
" hasLaneBeyond=" << hasLaneBeyond
3168 remaining > overtakeDist
3170 && (!checkCurrent || !checkOverTakeRight || !right)
3179 latDist = latLaneDist;
3180 mustOvertake =
true;
3181#ifdef DEBUG_STRATEGIC_CHANGE
3183 std::cout <<
" veh=" <<
myVehicle.
getID() <<
" overtake stopped leader=" << leader.first->getID()
3184 <<
" newCurrentDist=" << currentDist
3185 <<
" overtakeDist=" << overtakeDist
3186 <<
" remaining=" << remaining
3202 mustOvertake =
true;
3203 if (i >= rightmost && i <= leftmost) {
3210 return mustOvertake;
3231 double& maneuverDist,
3267 const double oldLatDist = latDist;
3268 const double oldManeuverDist = maneuverDist;
3273 const double halfWidth =
getWidth() * 0.5;
3279 double surplusGapRight = oldCenter - halfWidth;
3280 double surplusGapLeft =
getLeftBorder(laneOffset != 0) - oldCenter - halfWidth;
3281 const bool stayInLane = (laneOffset == 0
3285 && (surplusGapLeft >= 0 && surplusGapRight >= 0)));
3288 std::swap(surplusGapLeft, surplusGapRight);
3290#ifdef DEBUG_KEEP_LATGAP
3292 std::cout <<
"\n " <<
SIMTIME <<
" keepLatGap() laneOffset=" << laneOffset
3293 <<
" latDist=" << latDist
3294 <<
" maneuverDist=" << maneuverDist
3298 <<
" gapFactor=" << gapFactor
3299 <<
" stayInLane=" << stayInLane <<
"\n"
3300 <<
" stayInEdge: surplusGapRight=" << surplusGapRight <<
" surplusGapLeft=" << surplusGapLeft <<
"\n";
3304 if (surplusGapLeft < 0 || surplusGapRight < 0) {
3314 if (laneOffset != 0) {
3317 updateGaps(neighLeaders, neighRight, oldCenter, gapFactor, surplusGapRight, surplusGapLeft,
true);
3318 updateGaps(neighFollowers, neighRight, oldCenter, gapFactor, surplusGapRight, surplusGapLeft,
true, netOverlap);
3320#ifdef DEBUG_KEEP_LATGAP
3322 std::cout <<
" minGapLat: surplusGapRight=" << surplusGapRight <<
" surplusGapLeft=" << surplusGapLeft <<
"\n"
3332 if (stayInLane || laneOffset == 1) {
3335 surplusGapRight =
MIN2(surplusGapRight,
MAX2(0.0, halfLaneWidth + posLat - halfWidth));
3336 physicalGapRight =
MIN2(physicalGapRight,
MAX2(0.0, halfLaneWidth + posLat - halfWidth));
3338 if (stayInLane || laneOffset == -1) {
3341 surplusGapLeft =
MIN2(surplusGapLeft,
MAX2(0.0, halfLaneWidth - posLat - halfWidth));
3342 physicalGapLeft =
MIN2(physicalGapLeft,
MAX2(0.0, halfLaneWidth - posLat - halfWidth));
3344#ifdef DEBUG_KEEP_LATGAP
3346 std::cout <<
" stayInLane: surplusGapRight=" << surplusGapRight <<
" surplusGapLeft=" << surplusGapLeft <<
"\n";
3350 if (surplusGapRight + surplusGapLeft < 0) {
3355 const double equalDeficit = 0.5 * (surplusGapLeft + surplusGapRight);
3356 if (surplusGapRight < surplusGapLeft) {
3358 const double delta =
MIN2(equalDeficit - surplusGapRight, physicalGapLeft);
3360 maneuverDist = delta;
3361#ifdef DEBUG_KEEP_LATGAP
3363 std::cout <<
" insufficient latSpace, move left: delta=" << delta <<
"\n";
3368 const double delta =
MIN2(equalDeficit - surplusGapLeft, physicalGapRight);
3370 maneuverDist = -delta;
3371#ifdef DEBUG_KEEP_LATGAP
3373 std::cout <<
" insufficient latSpace, move right: delta=" << delta <<
"\n";
3379 latDist =
MAX2(
MIN2(latDist, surplusGapLeft), -surplusGapRight);
3380 maneuverDist =
MAX2(
MIN2(maneuverDist, surplusGapLeft), -surplusGapRight);
3381 if ((state &
LCA_KEEPRIGHT) != 0 && maneuverDist != oldManeuverDist) {
3383 latDist = oldLatDist;
3384 maneuverDist = oldManeuverDist;
3386#ifdef DEBUG_KEEP_LATGAP
3388 std::cout <<
" adapted latDist=" << latDist <<
" maneuverDist=" << maneuverDist <<
" (old=" << oldLatDist <<
")\n";
3403#ifdef DEBUG_KEEP_LATGAP
3405 std::cout <<
" traci influenced latDist=" << latDist <<
"\n";
3411 const bool traciChange = ((state | traciState) &
LCA_TRACI) != 0;
3412 if (nonSublaneChange && !traciChange) {
3414#ifdef DEBUG_KEEP_LATGAP
3416 std::cout <<
" wanted changeToLeft oldLatDist=" << oldLatDist <<
", blocked latGap changeToRight\n";
3419 latDist = oldLatDist;
3422#ifdef DEBUG_KEEP_LATGAP
3424 std::cout <<
" wanted changeToRight oldLatDist=" << oldLatDist <<
", blocked latGap changeToLeft\n";
3427 latDist = oldLatDist;
3437#ifdef DEBUG_KEEP_LATGAP
3439 std::cout <<
" latDistUpdated=" << latDist <<
" oldLatDist=" << oldLatDist <<
"\n";
3442 blocked =
checkBlocking(neighLane, latDist, maneuverDist, laneOffset, leaders, followers, blockers, neighLeaders, neighFollowers, neighBlockers,
nullptr,
nullptr, nonSublaneChange);
3445 state = (state & ~LCA_STAY);
3456#if defined(DEBUG_KEEP_LATGAP) || defined(DEBUG_STATE)
3458 std::cout <<
" latDist2=" << latDist
3472 double& surplusGapRight,
double& surplusGapLeft,
3473 bool saveMinGap,
double netOverlap,
3475 std::vector<CLeaderDist>* collectBlockers) {
3477 const double halfWidth =
getWidth() * 0.5 + NUMERICAL_EPS;
3480 if (others[i].first != 0 && others[i].second <= 0
3482 && (netOverlap == 0 || others[i].second + others[i].first->getVehicleType().getMinGap() < netOverlap)) {
3486 double foeRight, foeLeft;
3488 const double foeCenter = foeRight + 0.5 * res;
3489 const double gap =
MIN2(fabs(foeRight - oldCenter), fabs(foeLeft - oldCenter)) - halfWidth;
3492 const double currentMinGap = desiredMinGap * gapFactor;
3503#if defined(DEBUG_BLOCKING) || defined(DEBUG_KEEP_LATGAP)
3505 std::cout <<
" updateGaps"
3507 <<
" foe=" << foe->
getID()
3508 <<
" foeRight=" << foeRight
3509 <<
" foeLeft=" << foeLeft
3510 <<
" oldCenter=" << oldCenter
3511 <<
" gap=" << others[i].second
3512 <<
" latgap=" << gap
3513 <<
" currentMinGap=" << currentMinGap
3514 <<
" surplusGapRight=" << surplusGapRight
3515 <<
" surplusGapLeft=" << surplusGapLeft
3523 if (foeCenter < oldCenter) {
3525 surplusGapRight =
MIN3(surplusGapRight, gap - currentMinGap,
MAX2(currentMinGap, gap - foeManeuverDist));
3528 surplusGapLeft =
MIN3(surplusGapLeft, gap - currentMinGap,
MAX2(currentMinGap, gap - foeManeuverDist));
3531 if (foeCenter < oldCenter) {
3532#if defined(DEBUG_BLOCKING) || defined(DEBUG_KEEP_LATGAP)
3534 std::cout <<
" new minimum rightGap=" << gap <<
"\n";
3539#if defined(DEBUG_BLOCKING) || defined(DEBUG_KEEP_LATGAP)
3541 std::cout <<
" new minimum leftGap=" << gap <<
"\n";
3547 if (collectBlockers !=
nullptr) {
3549 if ((foeCenter < oldCenter && latDist < 0 && gap < (desiredMinGap - latDist))
3550 || (foeCenter > oldCenter && latDist > 0 && gap < (desiredMinGap + latDist))) {
3551 collectBlockers->push_back(others[i]);
3568 int currentDirection =
mySpeedLat >= 0 ? 1 : -1;
3569 int directionWish = latDist >= 0 ? 1 : -1;
3576 maxSpeedLat =
MIN2(maxSpeedLat, speedBound);
3580 maxSpeedLat =
MAX2(maxSpeedLat, speedBound);
3589 accelLat =
MAX2(accelLat, 2 * edgeOverlap);
3590 maxSpeedLat =
MAX2(maxSpeedLat, edgeOverlap);
3593#ifdef DEBUG_MANEUVER
3597 <<
" computeSpeedLat()"
3598 <<
" latDist=" << latDist
3599 <<
" maneuverDist=" << maneuverDist
3600 <<
" urgent=" << urgent
3602 <<
" currentDirection=" << currentDirection
3603 <<
" directionWish=" << directionWish
3605 <<
" maxSpeedLat=" << maxSpeedLat
3611 if (directionWish == 1) {
3625 if (maneuverDist * latDist > 0) {
3626 maneuverDist = fullLatDist;
3629#ifdef DEBUG_MANEUVER
3633 <<
" fullLatDist=" << fullLatDist
3634 <<
" speedAccel=" << speedAccel
3635 <<
" speedDecel=" << speedDecel
3636 <<
" speedBound=" << speedBound
3640 if (speedDecel * speedAccel <= 0 && (
3642 (latDist >= 0 && speedAccel >= speedBound && speedBound >= speedDecel)
3643 || (latDist <= 0 && speedAccel <= speedBound && speedBound <= speedDecel))) {
3645#ifdef DEBUG_MANEUVER
3647 std::cout <<
" computeSpeedLat a)\n";
3654#ifdef DEBUG_MANEUVER
3656 std::cout <<
" computeSpeedLat b)\n";
3663 if ((fabs(minDistAccel) < fabs(fullLatDist)) || (fabs(minDistAccel - fullLatDist) < NUMERICAL_EPS)) {
3664#ifdef DEBUG_MANEUVER
3666 std::cout <<
" computeSpeedLat c)\n";
3671#ifdef DEBUG_MANEUVER
3673 std::cout <<
" minDistAccel=" << minDistAccel <<
"\n";
3678 if ((fabs(minDistCurrent) < fabs(fullLatDist)) || (fabs(minDistCurrent - fullLatDist) < NUMERICAL_EPS)) {
3679#ifdef DEBUG_MANEUVER
3681 std::cout <<
" computeSpeedLat d)\n";
3688#ifdef DEBUG_MANEUVER
3690 std::cout <<
" computeSpeedLat e)\n";
3702#ifdef DEBUG_MANEUVER
3704 std::cout <<
" rightDanger speedLat=" << speedLat <<
"\n";
3709#ifdef DEBUG_MANEUVER
3711 std::cout <<
" leftDanger speedLat=" << speedLat <<
"\n";
3726 const bool indirect = turnInfo.second ==
nullptr ? false : turnInfo.second->isIndirect();
3759 double maneuverDist) {
3762 double secondsToLeaveLane;
3772#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3788 double nextLeftSpace;
3789 if (nextActionStepSpeed > 0.) {
3804#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3808 <<
" avoidArrivalSpeed=" << avoidArrivalSpeed
3811 <<
"\n nextLeftSpace=" << nextLeftSpace
3812 <<
" nextActionStepSpeed=" << nextActionStepSpeed
3813 <<
" nextActionStepRemainingSeconds=" << secondsToLeaveLane - timeTillActionStep
3823#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3827 <<
" secondsToLeave=" << secondsToLeaveLane
3849 const double vehWidth =
getWidth();
3851 const double leftVehSide = rightVehSide + vehWidth;
3852 const double rightVehSideDest = rightVehSide + latDist;
3853 const double leftVehSideDest = leftVehSide + latDist;
3854#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3856 std::cout <<
" commitFollowSpeed"
3857 <<
" latDist=" << latDist
3858 <<
" foeOffset=" << foeOffset
3859 <<
" vehRight=" << rightVehSide
3860 <<
" vehLeft=" << leftVehSide
3861 <<
" destRight=" << rightVehSideDest
3862 <<
" destLeft=" << leftVehSideDest
3868 if (vehDist.first != 0) {
3869 const MSVehicle* leader = vehDist.first;
3871 double foeRight, foeLeft;
3873#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3875 std::cout <<
" foe=" << vehDist.first->getID()
3876 <<
" gap=" << vehDist.second
3878 <<
" foeRight=" << foeRight
3879 <<
" foeLeft=" << foeLeft
3880 <<
" overlapBefore=" <<
overlap(rightVehSide, leftVehSide, foeRight, foeLeft)
3881 <<
" overlapDest=" <<
overlap(rightVehSideDest, leftVehSideDest, foeRight, foeLeft)
3885 if (
overlap(rightVehSideDest, leftVehSideDest, foeRight, foeLeft)) {
3889 speed =
MIN2(speed, vSafe);
3890#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3892 std::cout <<
" case1 vsafe=" << vSafe <<
" speed=" << speed <<
"\n";
3895 }
else if (
overlap(rightVehSide, leftVehSide, foeRight, foeLeft)) {
3900 speed =
MIN2(speed, vSafe);
3901#if defined(DEBUG_MANEUVER) || defined(DEBUG_COMMITTED_SPEED)
3903 std::cout <<
" case2 vsafe=" << vSafe <<
" speed=" << speed <<
"\n";
3923 return myOppositeParam <= 0 ? std::numeric_limits<double>::max() : 1 /
myOppositeParam;
3982 }
else if (key ==
"speedGainProbabilityRight") {
3984 }
else if (key ==
"speedGainProbabilityLeft") {
3986 }
else if (key ==
"keepRightProbability") {
3988 }
else if (key ==
"lookAheadSpeed") {
3990 }
else if (key ==
"sigmaState") {
3993 }
else if (key ==
"speedGainRP") {
3995 }
else if (key ==
"speedGainLP") {
3997 }
else if (key ==
"keepRightP") {
4067 }
else if (key ==
"speedGainProbabilityRight") {
4069 }
else if (key ==
"speedGainProbabilityLeft") {
4071 }
else if (key ==
"keepRightProbability") {
4073 }
else if (key ==
"lookAheadSpeed") {
4075 }
else if (key ==
"sigmaState") {
4089 const std::pair<MSVehicle*, double>& leader,
4090 const std::pair<MSVehicle*, double>& follower,
4091 const std::pair<MSVehicle*, double>& neighLead,
4092 const std::pair<MSVehicle*, double>& neighFollow,
4094 const std::vector<MSVehicle::LaneQ>& preb,
4100#ifdef DEBUG_WANTSCHANGE
4102 std::cout <<
"\nWANTS_CHANGE\n" <<
SIMTIME
4106 <<
" neigh=" << neighLane.
getID()
4110 <<
" considerChangeTo=" << (laneOffset == -1 ?
"right" :
"left")
4124 double maneuverDist;
4127 leaders, followers, blockers,
4128 neighLeaders, neighFollowers, neighBlockers,
4130 lastBlocked, firstBlocked, latDist, maneuverDist, blocked);
4134 result &= ~LCA_SUBLANE;
4135 result |=
getLCA(result, latDist);
4137#if defined(DEBUG_WANTSCHANGE) || defined(DEBUG_STATE)
4142 <<
" wantsChangeTo=" << (laneOffset == -1 ?
"right" :
"left")
4143 << ((result &
LCA_URGENT) ?
" (urgent)" :
"")
4149 << ((result &
LCA_TRACI) ?
" (traci)" :
"")
4201 std::cout <<
SIMTIME <<
" veh=" <<
myVehicle.
getID() <<
" bgap=" << brakeGap <<
" maneuverDist=" << maneuverDist
#define HELP_DECEL_FACTOR
#define LOOK_AHEAD_MIN_SPEED
#define LCA_RIGHT_IMPATIENCE
#define REACT_TO_STOPPED_DISTANCE
#define RELGAIN_NORMALIZATION_MIN_SPEED
#define CUT_IN_LEFT_SPEED_THRESHOLD
#define MAX_ONRAMP_LENGTH
#define LOOK_AHEAD_SPEED_MEMORY
#define ARRIVALPOS_LAT_THRESHOLD
#define SPEEDGAIN_MEMORY_FACTOR
#define LOOK_AHEAD_MIN_SPEED
#define SPEEDGAIN_DECAY_FACTOR
#define LATGAP_SPEED_THRESHOLD
#define GAIN_PERCEPTION_THRESHOLD
#define LATGAP_SPEED_THRESHOLD2
std::pair< const MSVehicle *, double > CLeaderDist
std::pair< const MSPerson *, double > PersonDist
LatAlignmentDefinition
Possible ways to choose the lateral alignment, i.e., how vehicles align themselves within their lane.
@ RIGHT
drive on the right side
@ GIVEN
The alignment as offset is given.
@ DEFAULT
No information given; use default.
@ LEFT
drive on the left side
@ ARBITRARY
maintain the current alignment
@ NICE
align with the closest sublane border
@ COMPACT
align with the rightmost sublane that allows keeping the current speed
@ CENTER
drive in the middle
@ SVC_EMERGENCY
public emergency vehicles
@ RIGHT
At the rightmost side of the lane.
@ GIVEN
The position is given.
@ DEFAULT
No information given; use default.
@ LEFT
At the leftmost side of the lane.
@ CENTER
At the center of the lane.
LinkDirection
The different directions a link between two lanes may take (or a stream between two edges)....
@ PARTLEFT
The link is a partial left direction.
@ RIGHT
The link is a (hard) right direction.
@ TURN
The link is a 180 degree turn.
@ LEFT
The link is a (hard) left direction.
@ STRAIGHT
The link is a straight direction.
@ TURN_LEFTHAND
The link is a 180 degree turn (left-hand network)
@ PARTRIGHT
The link is a partial right direction.
@ NODIR
The link has no direction (is a dead end link)
LaneChangeAction
The state of a vehicle's lane-change behavior.
@ LCA_BLOCKED_LEFT
blocked left
@ LCA_KEEPRIGHT
The action is due to the default of keeping right "Rechtsfahrgebot".
@ LCA_BLOCKED
blocked in all directions
@ LCA_URGENT
The action is urgent (to be defined by lc-model)
@ LCA_BLOCKED_BY_RIGHT_LEADER
The vehicle is blocked by right leader.
@ LCA_STAY
Needs to stay on the current lane.
@ LCA_SUBLANE
used by the sublane model
@ LCA_BLOCKED_BY_LEADER
blocked by leader
@ LCA_BLOCKED_BY_LEFT_FOLLOWER
The vehicle is blocked by left follower.
@ LCA_AMBLOCKINGFOLLOWER_DONTBRAKE
@ LCA_COOPERATIVE
The action is done to help someone else.
@ LCA_OVERLAPPING
The vehicle is blocked being overlapping.
@ LCA_LEFT
Wants go to the left.
@ LCA_BLOCKED_RIGHT
blocked right
@ LCA_BLOCKED_BY_RIGHT_FOLLOWER
The vehicle is blocked by right follower.
@ LCA_STRATEGIC
The action is needed to follow the route (navigational lc)
@ LCA_AMBACKBLOCKER_STANDING
@ LCA_CHANGE_REASONS
reasons of lane change
@ LCA_TRACI
The action is due to a TraCI request.
@ LCA_SPEEDGAIN
The action is due to the wish to be faster (tactical lc)
@ LCA_WANTS_LANECHANGE
lane can change
@ LCA_RIGHT
Wants go to the right.
@ LCA_BLOCKED_BY_FOLLOWER
blocker by follower
@ LCA_BLOCKED_BY_LEFT_LEADER
@ SUMO_ATTR_LCA_COOPERATIVE_SPEED
@ SUMO_ATTR_LCA_ASSERTIVE
@ SUMO_ATTR_LCA_LANE_DISCIPLINE
@ SUMO_ATTR_LCA_TURN_ALIGNMENT_DISTANCE
@ SUMO_ATTR_LCA_LOOKAHEADLEFT
@ SUMO_ATTR_LCA_SPEEDGAIN_PARAM
@ SUMO_ATTR_LCA_MAXDISTLATSTANDING
@ SUMO_ATTR_LCA_IMPATIENCE
@ SUMO_ATTR_LCA_COOPERATIVE_ROUNDABOUT
@ SUMO_ATTR_LCA_SPEEDGAIN_LOOKAHEAD
@ SUMO_ATTR_LCA_MAXSPEEDLATFACTOR
@ SUMO_ATTR_LCA_MAXSPEEDLATSTANDING
@ SUMO_ATTR_LCA_KEEPRIGHT_PARAM
@ SUMO_ATTR_LCA_COOPERATIVE_PARAM
@ SUMO_ATTR_LCA_OPPOSITE_PARAM
@ SUMO_ATTR_LCA_SPEEDGAIN_REMAIN_TIME
@ SUMO_ATTR_LCA_OVERTAKE_DELTASPEED_FACTOR
@ SUMO_ATTR_LCA_SUBLANE_PARAM
@ SUMO_ATTR_LCA_ACCEL_LAT
@ SUMO_ATTR_LCA_STRATEGIC_PARAM
@ SUMO_ATTR_LCA_KEEPRIGHT_ACCEPTANCE_TIME
@ SUMO_ATTR_LCA_TIME_TO_IMPATIENCE
@ SUMO_ATTR_LCA_SPEEDGAINRIGHT
int gPrecision
the precision for floating point outputs
const double SUMO_const_laneWidth
const double SUMO_const_haltingSpeed
the speed threshold at which vehicles are considered as halting
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
A class responsible for exchanging messages between cars involved in lane-change interaction.
Interface for lane-change models.
double getForwardPos() const
get vehicle position relative to the forward direction lane
virtual double getExtraReservation(int bestLaneOffset, double neighExtraDist=0) const
bool hasBlueLight() const
double getPreviousManeuverDist() const
virtual void setOwnState(const int state)
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.
virtual void prepareStep()
MSLane * getShadowLane() const
Returns the lane the vehicle's shadow is on during continuous/sublane lane change.
double myLastLateralGapRight
double myCommittedSpeed
the speed when committing to a change maneuver
virtual LatAlignmentDefinition getDesiredAlignment() const
static const double NO_NEIGHBOR
double myMaxDistLatStanding
static bool myAllowOvertakingRight
whether overtaking on the right is permitted
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 ...
int & getCanceledState(const int dir)
double myMaxSpeedLatFactor
const LaneChangeModel myModel
the type of this model
bool cancelRequest(int state, int laneOffset)
whether the influencer cancels the given request
std::vector< std::pair< double, bool > > myLCAccelerationAdvices
virtual bool avoidOvertakeRight(const MSVehicle *const neighLeader, const bool allowProb=false) const
double getMaxSpeedLat2() const
return the max of maxSpeedLat and lcMaxSpeedLatStanding
const MSCFModel & getCarFollowModel() const
The vehicle's car following model.
double mySpeedLat
the current lateral speed
double myMaxSpeedLatStanding
MSVehicle & myVehicle
The vehicle this lane-changer belongs to.
double myLastLateralGapLeft
the minimum lateral gaps to other vehicles that were found when last changing to the left and right
virtual bool debugVehicle() const
whether the current vehicles shall be debugged
virtual double getArrivalPos() const
Returns this vehicle's desired arrivalPos for its current route (may change on reroute)
const SUMOVehicleParameter & getParameter() const
Returns the vehicle's parameter (including departure definition)
double getLength() const
Returns the vehicle's length.
const MSEdge * getEdge() const
Returns the edge the vehicle is currently at.
double getWaitingSeconds() const
Returns the number of seconds waited (speed was lesser than 0.1m/s)
const MSStop & getNextStop() const
SUMOVehicleClass getVClass() const
Returns the vehicle's access class.
const MSRoute & getRoute() const
Returns the current route.
const MSVehicleType & getVehicleType() const
Returns the vehicle's type definition.
The car-following model abstraction.
virtual double maxNextSpeed(double speed, const MSVehicle *const veh) const
Returns the maximum speed given the current speed.
static double gapExtrapolation(const double duration, const double currentGap, double v1, double v2, double a1=0, double a2=0, const double maxV1=std::numeric_limits< double >::max(), const double maxV2=std::numeric_limits< double >::max())
return the resulting gap if, starting with gap currentGap, two vehicles continue with constant accele...
virtual double minNextSpeedEmergency(double speed, const MSVehicle *const veh=0) const
Returns the minimum speed after emergency braking, given the current speed (depends on the numerical ...
virtual double followSpeedTransient(double duration, const MSVehicle *const veh, double speed, double gap2pred, double predSpeed, double predMaxDecel) const
Computes the vehicle's follow speed that avoids a collision for the given amount of time.
double getEmergencyDecel() const
Get the vehicle type's maximal physically possible deceleration [m/s^2].
static double brakeGapEuler(const double speed, const double decel, const double headwayTime)
static double avoidArrivalAccel(double dist, double time, double speed, double maxDecel)
Computes the acceleration needed to arrive not before the given time.
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 void setMaxDecel(double decel)
Sets a new value for maximal comfortable deceleration [m/s^2].
@ LANE_CHANGE
the return value is used for lane change calculations
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 getMaxAccel() const
Get the vehicle type's maximum acceleration [m/s^2].
double brakeGap(const double speed) const
Returns the distance the vehicle needs to halt including driver's reaction time tau (i....
double getMaxDecel() const
Get the vehicle type's maximal comfortable deceleration [m/s^2].
static double estimateArrivalTime(double dist, double speed, double maxSpeed, double accel)
Computes the time needed to travel a distance dist given an initial speed and constant acceleration....
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)
A road/street connecting two junctions.
const std::set< MSTransportable *, ComparatorNumericalIdLess > & getPersons() const
Returns this edge's persons set.
const std::vector< MSLane * > & getLanes() const
Returns this edge's lanes.
double getInternalFollowingLengthTo(const MSEdge *followerAfterInternal, SUMOVehicleClass vClass) const
returns the length of all internal edges on the junction until reaching the non-internal edge followe...
bool canChangeToOpposite() const
whether this edge allows changing to the opposite direction edge
bool isInternal() const
return whether this edge is an internal edge
double getWidth() const
Returns the edges's width (sum over all lanes)
const std::vector< double > getSubLaneSides() const
Returns the right side offsets of this edge's sublanes.
static double gLateralResolution
static bool gSemiImplicitEulerUpdate
static bool gLefthand
Whether lefthand-drive is being simulated.
static bool canSaveBlockerLength(const MSVehicle &veh, double requested, double leftSpace)
static double getSpeedPreservingSecureGap(const MSVehicle &leader, const MSVehicle &follower, double currentGap, double leaderPlannedSpeed)
static double getRoundaboutDistBonus(const MSVehicle &veh, double bonusParam, const MSVehicle::LaneQ &curr, const MSVehicle::LaneQ &neigh, const MSVehicle::LaneQ &best)
static bool updateBlockerLength(const MSVehicle &veh, MSVehicle *blocker, int lcaCounter, double leftSpace, bool reliefConnection, double &leadingBlockerLength)
static bool divergentRoute(const MSVehicle &v1, const MSVehicle &v2)
return whether the vehicles are on the same junction but on divergent paths
double mySafeLatDistRight
the lateral distance the vehicle can safely move in the currently considered direction
static bool overlap(double right, double left, double right2, double left2)
return whether the given intervals overlap
double _patchSpeed(double min, const double wanted, double max, const MSCFModel &cfModel)
double informLeaders(int blocked, int dir, const std::vector< CLeaderDist > &blockers, double remainingSeconds)
void commitManoeuvre(int blocked, int blockedFully, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &neighLeaders, const MSLane &neighLane, double maneuverDist)
commit to lane change maneuver potentially overriding safe speed
std::set< const MSVehicle * > myCFRelated
set of vehicles that are in a car-following relationship with ego (leader of followers)
void prepareStep() override
double myKeepRightProbability
double commitFollowSpeed(double speed, double latDist, double secondsToLeaveLane, const MSLeaderDistanceInfo &leaders, double foeOffset) const
compute speed when committing to an urgent change that is safe in regard to leading vehicles
double getLeftBorder(bool checkOpposite=true) const
return current edge width optionally extended by opposite direction lane width
double myChangeProbThresholdRight
double informLeader(int blocked, int dir, const CLeaderDist &neighLead, double remainingSeconds)
MSLCM_SL2015(MSVehicle &v)
int computeSublaneShift(const MSEdge *prevEdge, const MSEdge *curEdge)
compute shift so that prevSublane + shift = newSublane
double patchSpeed(const double min, const double wanted, const double max, const MSCFModel &cfModel) override
Called to adapt the speed in order to allow a lane change. It uses information on LC-related desired ...
double getSafetyFactor() const override
return factor for modifying the safety constraints of the car-following model
double myCooperativeSpeed
double computeSpeedLat(double latDist, double &maneuverDist, bool urgent) const override
decides the next lateral speed depending on the remaining lane change distance to be covered and upda...
std::vector< double > myExpectedSublaneSpeeds
expected travel speeds on all sublanes on the current edge(!)
double getWidth() const
return the width of this vehicle (padded for numerical stability)
bool myCanChangeFully
whether the current lane changing maneuver can be finished in a single step
int wantsChange(int laneOffset, MSAbstractLaneChangeModel::MSLCMessager &msgPass, int blocked, const std::pair< MSVehicle *, double > &leader, const std::pair< MSVehicle *, double > &follower, const std::pair< MSVehicle *, double > &neighLead, const std::pair< MSVehicle *, double > &neighFollow, const MSLane &neighLane, const std::vector< MSVehicle::LaneQ > &preb, MSVehicle **lastBlocked, MSVehicle **firstBlocked) override
Called to examine whether the vehicle wants to change using the given laneOffset (this is a wrapper a...
bool mustOvertakeStopped(bool checkCurrent, const MSLane &neighLane, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &neighLead, double posOnLane, double neighDist, bool right, double latLaneDist, double ¤tDist, double &latDist)
bool outsideEdge() const
whether the ego vehicle is driving outside edgebounds
bool myDontBrake
flag to prevent speed adaptation by slowing down
std::string getParameter(const std::string &key) const override
try to retrieve the given parameter from this device. Throw exception for unsupported key
bool wantsKeepRight(double keepRightProb) const
check against thresholds
double forecastAverageSpeed(double vSafe, double vMax, double gap, double vLeader) const
estimate average speed over mySpeedGainLookahead time
int checkStrategicChange(int ret, const MSLane &neighLane, int laneOffset, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &neighLeaders, const MSVehicle::LaneQ &curr, const MSVehicle::LaneQ &neigh, const MSVehicle::LaneQ &best, int bestLaneOffset, bool changeToBest, double ¤tDist, double neighDist, double laDist, double roundaboutBonus, double latLaneDist, bool checkOpposite, double &latDist)
compute strategic lane change actions TODO: Better documentation, refs #2
void updateCFRelated(const MSLeaderDistanceInfo &vehicles, double foeOffset, bool leaders)
find leaders/followers that are already in a car-following relationship with ego
bool debugVehicle() const override
whether the current vehicles shall be debugged
int wantsChangeSublane(int laneOffset, LaneChangeAction alternatives, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &followers, const MSLeaderDistanceInfo &blockers, const MSLeaderDistanceInfo &neighLeaders, const MSLeaderDistanceInfo &neighFollowers, const MSLeaderDistanceInfo &neighBlockers, const MSLane &neighLane, const std::vector< MSVehicle::LaneQ > &preb, MSVehicle **lastBlocked, MSVehicle **firstBlocked, double &latDist, double &maneuverDist, int &blocked) override
Called to examine whether the vehicle wants to change with the given laneOffset (using the sublane mo...
LatAlignmentDefinition getDesiredAlignment() const override
double mySpeedGainProbabilityRight
a value for tracking the probability that a change to the right is beneficial
int slowDownForBlocked(MSVehicle **blocked, int state)
compute useful slowdowns for blocked vehicles
void initDerivedParameters()
init cached parameters derived directly from model parameters
int keepLatGap(int state, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &followers, const MSLeaderDistanceInfo &blockers, const MSLeaderDistanceInfo &neighLeaders, const MSLeaderDistanceInfo &neighFollowers, const MSLeaderDistanceInfo &neighBlockers, const MSLane &neighLane, int laneOffset, double &latDist, double &maneuverDist, int &blocked)
check whether lateral gap requirements are met override the current maneuver if necessary
bool tieBrakeLeader(const MSVehicle *veh) const
bool currentDistAllows(double dist, int laneOffset, double lookForwardDist)
CLeaderDist getLongest(const MSLeaderDistanceInfo &ldi) const
get the longest vehicle in the given info
double myCooperativeParam
double getNeighRight(const MSLane &neighLane) const
return the right offset of the neighboring lane relative to the current edge
double computeSpeedGain(double latDistSublane, double defaultNextSpeed) const
compute speedGain when moving by the given amount
double emergencySpeedLat(double speedLat) const
avoid unsafe lateral speed (overruling lcAccelLat)
double myKeepRightAcceptanceTime
void updateGaps(const MSLeaderDistanceInfo &others, double foeOffset, double oldCenter, double gapFactor, double &surplusGapRight, double &surplusGapLeft, bool saveMinGap=false, double netOverlap=0, double latDist=0, std::vector< CLeaderDist > *collectBlockers=0)
check remaining lateral gaps for the given foe vehicles and optionally update minimum lateral gaps
virtual void updateSafeLatDist(const double travelledLatDist) override
Updates the value of safe lateral distances (mySafeLatDistLeft and mySafeLatDistRight) during maneuve...
const MSEdge * myLastEdge
expected travel speeds on all sublanes on the current edge(!)
double getOppositeSafetyFactor() const override
return factor for modifying the safety constraints for opposite-diretction overtaking of the car-foll...
StateAndDist decideDirection(StateAndDist sd1, StateAndDist sd2) const override
decide in which direction to move in case both directions are desirable
std::pair< double, int > Info
information regarding save velocity (unused) and state flags of the ego vehicle
void msg(const CLeaderDist &cld, double speed, int state)
send a speed recommendation to the given vehicle
double mySpeedGainRemainTime
int checkBlocking(const MSLane &neighLane, double &latDist, double maneuverDist, int laneOffset, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &followers, const MSLeaderDistanceInfo &blockers, const MSLeaderDistanceInfo &neighLeaders, const MSLeaderDistanceInfo &neighFollowers, const MSLeaderDistanceInfo &neighBlockers, std::vector< CLeaderDist > *collectLeadBlockers=0, std::vector< CLeaderDist > *collectFollowBlockers=0, bool keepLatGapManeuver=false, double gapFactor=0, int *retBlockedFully=0)
restrict latDist to permissible speed and determine blocking state depending on that distance
double getVehicleCenter() const
return vehicle position relative to the current edge (extend by another virtual lane for opposite-dir...
int _wantsChangeSublane(int laneOffset, LaneChangeAction alternatives, const MSLeaderDistanceInfo &leaders, const MSLeaderDistanceInfo &followers, const MSLeaderDistanceInfo &blockers, const MSLeaderDistanceInfo &neighLeaders, const MSLeaderDistanceInfo &neighFollowers, const MSLeaderDistanceInfo &neighBlockers, const MSLane &neighLane, const std::vector< MSVehicle::LaneQ > &preb, MSVehicle **lastBlocked, MSVehicle **firstBlocked, double &latDist, double &maneuverDist, int &blocked)
helper function for doing the actual work
double getLateralDrift()
get lateral drift for the current step
double computeGapFactor(int state) const
compute the gap factor for the given state
double getPosLat()
get lateral position of this vehicle
bool preventSliding(double maneuverDist) const
bool isBidi(const MSLane *lane) const
check whether lane is an upcoming bidi lane
void * inform(void *info, MSVehicle *sender) override
void informFollower(int blocked, int dir, const CLeaderDist &neighFollow, double remainingSeconds, double plannedSpeed)
decide whether we will try cut in before the follower or allow to be overtaken
void setParameter(const std::string &key, const std::string &value) override
try to set the given parameter for this laneChangeModel. Throw exception for unsupported key
bool saveBlockerLength(double length, double foeLeftSpace) override
reserve space at the end of the lane to avoid dead locks
double myOvertakeDeltaSpeedFactor
double myTurnAlignmentDist
double myLeadingBlockerLength
void setOwnState(const int state) override
int checkBlockingVehicles(const MSVehicle *ego, const MSLeaderDistanceInfo &vehicles, int laneOffset, double latDist, double foeOffset, bool leaders, double &safeLatGapRight, double &safeLatGapLeft, std::vector< CLeaderDist > *collectBlockers=0) const
check whether any of the vehicles overlaps with ego
void informFollowers(int blocked, int dir, const std::vector< CLeaderDist > &blockers, double remainingSeconds, double plannedSpeed)
call informFollower for multiple followers
static const MSVehicle * getStopped(const MSLeaderDistanceInfo &ldi)
get a stopped vehicle in the given info
double mySpeedGainLookahead
double mySpeedLossProbThreshold
void resetState() override
double mySpeedGainProbabilityLeft
a value for tracking the probability that a change to the left is beneficial
static LaneChangeAction getLCA(int state, double latDist)
compute lane change action from desired lateral distance
double myChangeProbThresholdLeft
void updateExpectedSublaneSpeeds(const MSLeaderDistanceInfo &ahead, int sublaneOffset, int laneIndex) override
update expected speeds for each sublane of the current edge
bool sublaneEnds(int i, const MSLane *next, double shift)
check whether the sublane continues on the next lane
bool currentDistDisallows(double dist, int laneOffset, double lookForwardDist)
double myTimeToImpatience
static int lowest_bit(int changeReason)
return the most important change reason
static CLeaderDist getSlowest(const MSLeaderDistanceInfo &ldi)
get the slowest vehicle in the given info
bool amBlockingFollowerPlusNB()
Representation of a lane in the micro simulation.
std::pair< const MSPerson *, double > nextBlocking(double minPos, double minRight, double maxLeft, double stopTime=0, bool bidi=false) const
This is just a wrapper around MSPModel::nextBlocking. You should always check using hasPedestrians be...
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.
const MSLink * getLinkTo(const MSLane *const) const
returns the link to the given lane or nullptr, if it is not connected
double getSpeedLimit() const
Returns the lane's maximum allowed speed.
double getLength() const
Returns the lane's length.
bool allowsVehicleClass(SUMOVehicleClass vclass) const
double getVehicleMaxSpeed(const SUMOTrafficObject *const veh) const
Returns the lane's maximum speed, given a vehicle's speed limit adaptation.
double getRightSideOnEdge() const
bool hasPedestrians() const
whether the lane has pedestrians on it
int getIndex() const
Returns the lane's index.
MSLane * getOpposite() const
return the neighboring opposite direction lane for lane changing or nullptr
MSLane * getBidiLane() const
retrieve bidirectional lane or nullptr
MSLane * getParallelOpposite() const
return the opposite direction lane of this lanes edge or nullptr
MSEdge & getEdge() const
Returns the lane's edge.
const MSLane * getNormalPredecessorLane() const
get normal lane leading to this internal lane, for normal lanes, the lane itself is returned
double getWidth() const
Returns the lane's width.
const std::vector< MSLink * > & getLinkCont() const
returns the container with all links !!!
int getRightmostSublane() const
saves leader/follower vehicles and their distances relative to an ego vehicle
virtual std::string toString() const
print a debugging representation
double getMinDistToStopped() const
return minimum distance to a stopped vehicle or max double
bool hasStoppedVehicle() const
whether a stopped vehicle is leader
void getSublaneBorders(int sublane, double latOffset, double &rightSide, double &leftSide) const
void getSubLanes(const MSVehicle *veh, double latOffset, int &rightmost, int &leftmost) const
const std::vector< const MSVehicle * > & getVehicles() const
double getLateralShift() const
return lateral shift that must be applied when passing this link
static MSNet * getInstance()
Returns the pointer to the unique instance of MSNet (singleton).
SUMOTime getCurrentTimeStep() const
Returns the current simulation step.
const MSEdge * getLastEdge() const
returns the destination edge
const MSLane * lane
The lane to stop at (microsim only)
double getLatDist() const
double changeRequestRemainingSeconds(const SUMOTime currentTime) const
Return the remaining number of seconds of the current laneTimeLine assuming one exists.
bool ignoreOverlap() const
Representation of a vehicle in the micro simulation.
double getRightSideOnEdge(const MSLane *lane=0) const
Get the vehicle's lateral position on the edge of the given lane (or its current edge if lane == 0)
double getLeftSideOnEdge(const MSLane *lane=0) const
Get the vehicle's lateral position on the edge of the given lane (or its current edge if lane == 0)
bool isActive() const
Returns whether the current simulation step is an action point for the vehicle.
const std::pair< double, const MSLink * > & getNextTurn()
Get the distance and direction of the next upcoming turn for the vehicle (within its look-ahead range...
MSAbstractLaneChangeModel & getLaneChangeModel()
double getLeftSideOnLane() const
Get the lateral position of the vehicles left side on the lane:
double getActionStepLengthSecs() const
Returns the vehicle's action step length in secs, i.e. the interval between two action points.
int influenceChangeDecision(int state)
allow TraCI to influence a lane change decision
double nextStopDist() const
return the distance to the next stop or doubleMax if there is none.
double getAcceleration() const
Returns the vehicle's acceleration in m/s (this is computed as the last step's mean acceleration in c...
const std::vector< MSLane * > & getBestLanesContinuation() const
Returns the best sequence of lanes to continue the route starting at myLane.
int getBestLaneOffset() const
double lateralDistanceToLane(const int offset) const
Get the minimal lateral distance required to move fully onto the lane at given offset.
const MSEdge * getCurrentEdge() const
Returns the edge the vehicle is currently at (possibly an internal edge or nullptr)
const MSLane * getLane() const
Returns the lane the vehicle is on.
double getLastStepDist() const
Get the distance the vehicle covered in the previous timestep.
Influencer & getInfluencer()
double getRightSideOnLane() const
Get the lateral position of the vehicles right side on the lane:
double getLateralPositionOnLane() const
Get the vehicle's lateral position on the lane.
double getSpeed() const
Returns the vehicle's current speed.
const std::vector< LaneQ > & getBestLanes() const
Returns the description of best lanes to use in order to continue the route.
const MSCFModel & getCarFollowModel() const
Returns the vehicle's car following model definition.
double getPositionOnLane() const
Get the vehicle's position along the lane.
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)
double getCenterOnEdge(const MSLane *lane=0) const
Get the vehicle's lateral position on the edge of the given lane (or its current edge if lane == 0)
double getLengthWithGap() const
Get vehicle's length including the minimum gap [m].
double getWidth() const
Get the width which vehicles of this class shall have when being drawn.
SUMOVehicleClass getVehicleClass() const
Get this vehicle type's vehicle class.
const LatAlignmentDefinition & getPreferredLateralAlignment() const
Get vehicle's preferred lateral alignment procedure.
double getMinGap() const
Get the free space in front of vehicles of this class.
double getMaxSpeedLat() const
Get vehicle's maximum lateral speed [m/s].
double getLength() const
Get vehicle's length [m].
double getPreferredLateralAlignmentOffset() const
Get vehicle's preferred lateral alignment offset (in m from center line)
static std::string getIDSecure(const T *obj, const std::string &fallBack="NULL")
get an identifier for Named-like object which may be Null
const std::string & getID() const
Returns the id.
void step(double dt)
evolve for a time step of length dt.
double arrivalPosLat
(optional) The lateral position the vehicle shall arrive on
ArrivalPosLatDefinition arrivalPosLatProcedure
Information how the vehicle shall choose the lateral arrival position.
static double toDouble(const std::string &sData)
converts a string into the double value described by it by calling the char-type converter
#define UNUSED_PARAMETER(x)
NLOHMANN_BASIC_JSON_TPL_DECLARATION void swap(nlohmann::NLOHMANN_BASIC_JSON_TPL &j1, nlohmann::NLOHMANN_BASIC_JSON_TPL &j2) noexcept(//NOLINT(readability-inconsistent-declaration-parameter-name) is_nothrow_move_constructible< nlohmann::NLOHMANN_BASIC_JSON_TPL >::value &&//NOLINT(misc-redundant-expression) is_nothrow_move_assignable< nlohmann::NLOHMANN_BASIC_JSON_TPL >::value)
exchanges the values of two JSON objects
bool sameDirection(const StateAndDist &other) const
A structure representing the best lanes for continuing the current route starting at 'lane'.
double length
The overall length which may be driven when using this lane without a lane change.
std::vector< MSLane * > bestContinuations
MSLane * lane
The described lane.
int bestLaneOffset
The (signed) number of lanes to be crossed to get to the lane which allows to continue the drive.
double occupation
The overall vehicle sum on consecutive lanes which can be passed without a lane change.