Line data Source code
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 : /****************************************************************************/
14 : /// @file MSLane.h
15 : /// @author Christian Roessel
16 : /// @author Daniel Krajzewicz
17 : /// @author Jakob Erdmann
18 : /// @author Christoph Sommer
19 : /// @author Tino Morenz
20 : /// @author Michael Behrisch
21 : /// @author Mario Krumnow
22 : /// @author Leonhard Luecken
23 : /// @date Mon, 12 Mar 2001
24 : ///
25 : // Representation of a lane in the micro simulation
26 : /****************************************************************************/
27 : #pragma once
28 : #include <config.h>
29 :
30 : #include <memory>
31 : #include <vector>
32 : #include <map>
33 : #include <deque>
34 : #include <cassert>
35 : #include <utils/common/Named.h>
36 : #include <utils/common/Parameterised.h>
37 : #include <utils/common/SUMOVehicleClass.h>
38 : #include <utils/vehicle/SUMOVehicle.h>
39 : #include <utils/common/NamedRTree.h>
40 : #include <utils/emissions/PollutantsInterface.h>
41 : #include <utils/geom/PositionVector.h>
42 : #include "MSGlobals.h"
43 : #include "MSLeaderInfo.h"
44 : #include "MSMoveReminder.h"
45 : #include "MSVehicle.h"
46 :
47 : #include <utils/foxtools/MFXSynchQue.h>
48 : #ifdef HAVE_FOX
49 : #include <utils/foxtools/MFXWorkerThread.h>
50 : #endif
51 : #include <utils/common/StopWatch.h>
52 :
53 :
54 : // ===========================================================================
55 : // class declarations
56 : // ===========================================================================
57 : class MSEdge;
58 : class MSBaseVehicle;
59 : class MSLaneChanger;
60 : class MSLink;
61 : class MSVehicleTransfer;
62 : class MSVehicleControl;
63 : class OutputDevice;
64 : class MSLeaderInfo;
65 : class MSJunction;
66 :
67 :
68 : // ===========================================================================
69 : // type definitions
70 : // ===========================================================================
71 : /// Coverage info
72 : typedef std::map<const MSLane*, std::pair<double, double> > LaneCoverageInfo;
73 :
74 : // ===========================================================================
75 : // class definitions
76 : // ===========================================================================
77 : /**
78 : * @class MSLane
79 : * @brief Representation of a lane in the micro simulation
80 : *
81 : * Class which represents a single lane. Somekind of the main class of the
82 : * simulation. Allows moving vehicles.
83 : */
84 : class MSLane : public Named, public Parameterised {
85 : public:
86 : class StoringVisitor {
87 : public:
88 : /// @brief Constructor
89 : StoringVisitor(std::set<const Named*>& objects, const PositionVector& shape,
90 : const double range, const int domain)
91 123236 : : myObjects(objects), myShape(shape), myRange(range), myDomain(domain) {}
92 :
93 : /// @brief Adds the given object to the container
94 : void add(const MSLane* const l) const;
95 :
96 : private:
97 : /// @brief The container
98 : std::set<const Named*>& myObjects;
99 : const PositionVector& myShape;
100 : const double myRange;
101 : const int myDomain;
102 :
103 : private:
104 : /// @brief invalidated copy constructor
105 : StoringVisitor(const StoringVisitor& src);
106 :
107 : /// @brief invalidated assignment operator
108 : StoringVisitor& operator=(const StoringVisitor& src);
109 : };
110 :
111 : /// needs access to myTmpVehicles (this maybe should be done via double-buffering!!!)
112 : friend class MSLaneChanger;
113 : friend class MSLaneChangerSublane;
114 :
115 : friend class MSQueueExport;
116 : friend class AnyVehicleIterator;
117 :
118 : /// Container for vehicles.
119 : typedef std::vector<MSVehicle*> VehCont;
120 :
121 : // TODO: Better documentation
122 : /// @brief AnyVehicleIterator is a structure, which manages the iteration through all vehicles on the lane,
123 : /// that may be of importance for the car-following dynamics along that lane. The relevant types of vehicles are:
124 : /// 1) vehicles with their front on the lane (myVehicles),
125 : /// 2) vehicles intersecting the lane but with front on another lane (myPartialVehicles)
126 : ///
127 : /// In the context of retrieving linkLeaders during lane changing a third group of vehicles is checked:
128 : /// 3) vehicles processed during lane changing (myTmpVehicles)
129 : class AnyVehicleIterator {
130 : public:
131 : AnyVehicleIterator(
132 : const MSLane* lane,
133 : int i1,
134 : int i2,
135 : int i3,
136 : const int i1End,
137 : const int i2End,
138 : const int i3End,
139 1593682580 : bool downstream = true) :
140 1593682580 : myLane(lane),
141 1593682580 : myI1(i1),
142 1593682580 : myI2(i2),
143 1593682580 : myI3(i3),
144 1593682580 : myI1End(i1End),
145 1593682580 : myI2End(i2End),
146 1593682580 : myI3End(i3End),
147 1593682580 : myDownstream(downstream),
148 1593682580 : myDirection(downstream ? 1 : -1) {
149 : }
150 :
151 : bool operator== (AnyVehicleIterator const& other) const {
152 : return (myI1 == other.myI1
153 11352122179 : && myI2 == other.myI2
154 : && myI3 == other.myI3
155 3246947377 : && myI1End == other.myI1End
156 : && myI2End == other.myI2End
157 1341908547 : && myI3End == other.myI3End);
158 : }
159 :
160 : bool operator!= (AnyVehicleIterator const& other) const {
161 : return !(*this == other);
162 : }
163 :
164 : const MSVehicle* operator->() {
165 : return **this;
166 : }
167 :
168 : const MSVehicle* operator*();
169 :
170 : AnyVehicleIterator& operator++();
171 :
172 : private:
173 : bool nextIsMyVehicles() const;
174 :
175 : /// @brief the lane that is being iterated
176 : const MSLane* myLane;
177 : /// @brief index for myVehicles
178 : int myI1;
179 : /// @brief index for myPartialVehicles
180 : int myI2;
181 : /// @brief index for myTmpVehicles
182 : int myI3;
183 : /// @brief end index for myVehicles
184 : int myI1End;
185 : /// @brief end index for myPartialVehicles
186 : int myI2End;
187 : /// @brief end index for myTmpVehicles
188 : int myI3End;
189 : /// @brief iteration direction
190 : bool myDownstream;
191 : /// @brief index delta
192 : int myDirection;
193 :
194 : };
195 :
196 :
197 : public:
198 : /** @enum ChangeRequest
199 : * @brief Requests set via TraCI
200 : */
201 : enum CollisionAction {
202 : COLLISION_ACTION_NONE,
203 : COLLISION_ACTION_WARN,
204 : COLLISION_ACTION_TELEPORT,
205 : COLLISION_ACTION_REMOVE
206 : };
207 :
208 : /** @brief Constructor
209 : *
210 : * @param[in] id The lane's id
211 : * @param[in] maxSpeed The speed allowed on this lane
212 : * @param[in] friction The friction of this lane
213 : * @param[in] length The lane's length
214 : * @param[in] edge The edge this lane belongs to
215 : * @param[in] numericalID The numerical id of the lane
216 : * @param[in] shape The shape of the lane
217 : * @param[in] width The width of the lane
218 : * @param[in] permissions Encoding of the Vehicle classes that may drive on this lane
219 : * @param[in] index The index of this lane within its parent edge
220 : * @param[in] isRampAccel Whether this lane is an acceleration lane
221 : * @see SUMOVehicleClass
222 : */
223 : MSLane(const std::string& id, double maxSpeed, double friction, double length, MSEdge* const edge,
224 : int numericalID, const PositionVector& shape, double width,
225 : SVCPermissions permissions,
226 : SVCPermissions changeLeft, SVCPermissions changeRight,
227 : int index, bool isRampAccel,
228 : const std::string& type,
229 : const PositionVector& outlineShape);
230 :
231 :
232 : /// @brief Destructor
233 : virtual ~MSLane();
234 :
235 : /// @brief returns the associated thread index
236 : inline int getThreadIndex() const {
237 : return myRNGIndex % MSGlobals::gNumSimThreads;
238 : }
239 :
240 : /// @brief returns the associated RNG index
241 : inline int getRNGIndex() const {
242 36081113 : return myRNGIndex;
243 : }
244 :
245 : /// @brief return the associated RNG
246 : SumoRNG* getRNG() const {
247 746915858 : return &myRNGs[myRNGIndex];
248 : }
249 :
250 : /// @brief return the number of RNGs
251 : static int getNumRNGs() {
252 1677 : return (int)myRNGs.size();
253 : }
254 :
255 : /// @brief save random number generator states to the given output device
256 : static void saveRNGStates(OutputDevice& out);
257 :
258 : /// @brief load random number generator state for the given rng index
259 : static void loadRNGState(int index, const std::string& state);
260 :
261 : /// @name Additional initialisation
262 : /// @{
263 :
264 : /** @brief Delayed initialization
265 : *
266 : * Not all lane-members are known at the time the lane is born, above all the pointers
267 : * to other lanes, so we have to add them later.
268 : *
269 : * @param[in] link An outgoing link
270 : */
271 : void addLink(MSLink* link);
272 :
273 : /** @brief Adds a neighbor to this lane
274 : *
275 : * @param[in] id The lane's id
276 : */
277 : void setOpposite(MSLane* oppositeLane);
278 :
279 : /** @brief Adds the (overlapping) reverse direction lane to this lane
280 : *
281 : * @param[in] id The lane's id
282 : */
283 : void setBidiLane(MSLane* bidyLane);
284 : ///@}
285 :
286 : /// @name Used by the GUI for secondary shape visualization
287 : /// @{
288 0 : virtual void addSecondaryShape(const PositionVector& /*shape*/) {}
289 :
290 1856 : virtual double getLengthGeometryFactor(bool /*secondaryShape*/) const {
291 1856 : return myLengthGeometryFactor;
292 : }
293 :
294 1632 : virtual const PositionVector& getShape(bool /*secondaryShape*/) const {
295 1632 : return myShape;
296 : }
297 : ///@}
298 :
299 195779 : virtual void updateMesoGUISegments() {}
300 :
301 : /// @name interaction with MSMoveReminder
302 : /// @{
303 :
304 : /** @brief Add a move-reminder to move-reminder container
305 : *
306 : * The move reminder will not be deleted by the lane.
307 : *
308 : * @param[in] rem The move reminder to add
309 : */
310 : virtual void addMoveReminder(MSMoveReminder* rem, bool addToVehicles = true);
311 :
312 :
313 : /** @brief Remove a move-reminder from move-reminder container
314 : *
315 : * The move reminder will not be deleted by the lane.
316 : * @param[in] rem The move reminder to remvoe
317 : */
318 : virtual void removeMoveReminder(MSMoveReminder* rem);
319 :
320 :
321 : /** @brief Return the list of this lane's move reminders
322 : * @return Previously added move reminder
323 : */
324 : inline const std::vector< MSMoveReminder* >& getMoveReminders() const {
325 : return myMoveReminders;
326 : }
327 : ///@}
328 :
329 :
330 :
331 : /// @name Vehicle insertion
332 : ///@{
333 :
334 : /** @brief Tries to insert the given vehicle
335 : *
336 : * The insertion position and speed are determined in dependence
337 : * to the vehicle's departure definition, first.
338 : *
339 : * Then, the vehicle is tried to be inserted into the lane
340 : * using these values by a call to "isInsertionSuccess". The result of
341 : * "isInsertionSuccess" is returned.
342 : *
343 : * @param[in] v The vehicle to insert
344 : * @return Whether the vehicle could be inserted
345 : * @see isInsertionSuccess
346 : * @see MSVehicle::getDepartureDefinition
347 : * @see MSVehicle::DepartArrivalDefinition
348 : */
349 : bool insertVehicle(MSVehicle& v);
350 :
351 :
352 : /** @brief Tries to insert the given vehicle with the given state (speed and pos)
353 : *
354 : * Checks whether the vehicle can be inserted at the given position with the
355 : * given speed so that no collisions with leader/follower occur and the speed
356 : * does not cause unexpected behaviour on consecutive lanes. Returns false
357 : * if the vehicle can not be inserted.
358 : *
359 : * If the insertion can take place, incorporateVehicle() is called and true is returned.
360 : *
361 : * @param[in] vehicle The vehicle to insert
362 : * @param[in] speed The speed with which it shall be inserted
363 : * @param[in] pos The position at which it shall be inserted
364 : * @param[in] posLat The lateral position at which it shall be inserted
365 : * @param[in] recheckNextLanes Forces patching the speed for not being too fast on next lanes
366 : * @param[in] notification The cause of insertion (i.e. departure, teleport, parking) defaults to departure
367 : * @return Whether the vehicle could be inserted
368 : * @see MSVehicle::enterLaneAtInsertion
369 : */
370 : bool isInsertionSuccess(MSVehicle* vehicle, double speed, double pos, double posLat,
371 : bool recheckNextLanes,
372 : MSMoveReminder::Notification notification);
373 :
374 : // XXX: Documentation?
375 : bool checkFailure(const MSVehicle* aVehicle, double& speed, double& dist, const double nspeed, const bool patchSpeed, const std::string errorMsg, InsertionCheck check) const;
376 :
377 : /** @brief inserts vehicle as close as possible to the last vehicle on this
378 : * lane (or at the end of the lane if there is no leader)
379 : */
380 : bool lastInsertion(MSVehicle& veh, double mspeed, double posLat, bool patchSpeed);
381 :
382 : /** @brief Tries to insert the given vehicle on any place
383 : *
384 : * @param[in] veh The vehicle to insert
385 : * @param[in] speed The maximum insertion speed
386 : * @param[in] notification The cause of insertion (i.e. departure, teleport, parking) defaults to departure
387 : * @return Whether the vehicle could be inserted
388 : */
389 : bool freeInsertion(MSVehicle& veh, double speed, double posLat,
390 : MSMoveReminder::Notification notification = MSMoveReminder::NOTIFICATION_DEPARTED);
391 :
392 :
393 : /** @brief Inserts the given vehicle at the given position
394 : *
395 : * No checks are done, vehicle insertion using this method may
396 : * generate collisions (possibly delayed).
397 : * @param[in] veh The vehicle to insert
398 : * @param[in] pos The position at which the vehicle shall be inserted
399 : * @param[in] notification The cause of insertion (i.e. departure, teleport, parking) defaults to departure
400 : * @param[in] posLat The lateral position at which the vehicle shall be inserted
401 : */
402 : void forceVehicleInsertion(MSVehicle* veh, double pos, MSMoveReminder::Notification notification, double posLat = 0);
403 : /// @}
404 :
405 :
406 :
407 : /// @name Handling vehicles lapping into several lanes (-> partial occupation)
408 : /// or which committed a maneuver that will lead them into another (sublane case -> maneuver reservations)
409 : /// @{
410 : /** @brief Sets the information about a vehicle lapping into this lane
411 : *
412 : * This vehicle is added to myVehicles and may be distinguished from regular
413 : * vehicles by the disparity between this lane and v->getLane()
414 : * @param[in] v The vehicle which laps into this lane
415 : * @return This lane's length
416 : */
417 : virtual double setPartialOccupation(MSVehicle* v);
418 :
419 : /** @brief Removes the information about a vehicle lapping into this lane
420 : * @param[in] v The vehicle which laps into this lane
421 : */
422 : virtual void resetPartialOccupation(MSVehicle* v);
423 :
424 : /** @brief Registers the lane change intentions (towards this lane) for the given vehicle
425 : */
426 : virtual void setManeuverReservation(MSVehicle* v);
427 :
428 : /** @brief Unregisters a vehicle, which previously registered for maneuvering into this lane
429 : * @param[in] v The vehicle
430 : */
431 : virtual void resetManeuverReservation(MSVehicle* v);
432 :
433 : /** @brief Returns the last vehicles on the lane
434 : *
435 : * The information about the last vehicles in this lanes in all sublanes
436 : * occupied by ego are
437 : * returned. Partial occupators are included
438 : * @param[in] ego The vehicle for which to restrict the returned leaderInfo
439 : * @param[in] minPos The minimum position from which to start search for leaders
440 : * @param[in] allowCached Whether the cached value may be used
441 : * @return Information about the last vehicles
442 : */
443 : const MSLeaderInfo getLastVehicleInformation(const MSVehicle* ego, double latOffset, double minPos = 0, bool allowCached = true) const;
444 :
445 : /// @brief analogue to getLastVehicleInformation but in the upstream direction
446 : const MSLeaderInfo getFirstVehicleInformation(const MSVehicle* ego, double latOffset, bool onlyFrontOnLane, double maxPos = std::numeric_limits<double>::max(), bool allowCached = true) const;
447 :
448 : /// @}
449 :
450 : /// @name Access to vehicles
451 : /// @{
452 :
453 : /** @brief Returns the number of vehicles on this lane (for which this lane
454 : * is responsible)
455 : * @return The number of vehicles with their front on this lane
456 : */
457 : int getVehicleNumber() const {
458 591942253 : return (int)myVehicles.size();
459 : }
460 :
461 : /** @brief Returns the number of vehicles on this lane (including partial
462 : * occupators)
463 : * @return The number of vehicles with intersecting this lane
464 : */
465 : int getVehicleNumberWithPartials() const {
466 74537497 : return (int)myVehicles.size() + (int)myPartialVehicles.size();
467 : }
468 :
469 : /** @brief Returns the number of vehicles partially on this lane (for which this lane
470 : * is not responsible)
471 : * @return The number of vehicles touching this lane but with their front on another lane
472 : */
473 : int getPartialVehicleNumber() const {
474 : return (int)myPartialVehicles.size();
475 : }
476 :
477 :
478 : /** @brief Returns the vehicles container; locks it for microsimulation
479 : *
480 : * Please note that it is necessary to release the vehicles container
481 : * afterwards using "releaseVehicles".
482 : * @return The vehicles on this lane
483 : */
484 3555597911 : virtual const VehCont& getVehiclesSecure() const {
485 3555597911 : return myVehicles;
486 : }
487 :
488 :
489 : /// @brief begin iterator for iterating over all vehicles touching this lane in downstream direction
490 : AnyVehicleIterator anyVehiclesBegin() const {
491 : return AnyVehicleIterator(this, 0, 0, 0,
492 1091928136 : (int)myVehicles.size(), (int)myPartialVehicles.size(), (int)myTmpVehicles.size(), true);
493 : }
494 :
495 : /// @brief end iterator for iterating over all vehicles touching this lane in downstream direction
496 : AnyVehicleIterator anyVehiclesEnd() const {
497 : return AnyVehicleIterator(this, (int)myVehicles.size(), (int)myPartialVehicles.size(), (int)myTmpVehicles.size(),
498 11269321819 : (int)myVehicles.size(), (int)myPartialVehicles.size(), (int)myTmpVehicles.size(), true);
499 : }
500 :
501 : /// @brief begin iterator for iterating over all vehicles touching this lane in upstream direction
502 : AnyVehicleIterator anyVehiclesUpstreamBegin() const {
503 : return AnyVehicleIterator(this, (int)myVehicles.size() - 1, (int)myPartialVehicles.size() - 1, (int)myTmpVehicles.size() - 1,
504 468332101 : -1, -1, -1, false);
505 : }
506 :
507 : /// @brief end iterator for iterating over all vehicles touching this lane in upstream direction
508 : AnyVehicleIterator anyVehiclesUpstreamEnd() const {
509 : return AnyVehicleIterator(this, -1, -1, -1, -1, -1, -1, false);
510 : }
511 :
512 : /** @brief Allows to use the container for microsimulation again
513 : */
514 3555597901 : virtual void releaseVehicles() const { }
515 : /// @}
516 :
517 :
518 :
519 : /// @name Atomar value getter
520 : /// @{
521 :
522 :
523 : /** @brief Returns this lane's numerical id
524 : * @return This lane's numerical id
525 : */
526 : inline int getNumericalID() const {
527 342913288 : return myNumericalID;
528 : }
529 :
530 :
531 : /** @brief Returns this lane's shape
532 : * @return This lane's shape
533 : */
534 : inline const PositionVector& getShape() const {
535 22028173 : return myShape;
536 : }
537 :
538 : /// @brief return shape.length() / myLength
539 : inline double getLengthGeometryFactor() const {
540 2655453 : return myLengthGeometryFactor;
541 : }
542 :
543 : /// @brief return whether this lane is an acceleration lane
544 : inline bool isAccelLane() const {
545 603477 : return myIsRampAccel;
546 : }
547 :
548 : /// @brief return the type of this lane
549 : const std::string& getLaneType() const {
550 : return myLaneType;
551 : }
552 :
553 : /* @brief fit the given lane position to a visibly suitable geometry position
554 : * (lane length might differ from geometry length) */
555 : inline double interpolateLanePosToGeometryPos(double lanePos) const {
556 8872029 : return lanePos * myLengthGeometryFactor;
557 : }
558 :
559 : /* @brief fit the given lane position to a visibly suitable geometry position
560 : * and return the coordinates */
561 : inline const Position geometryPositionAtOffset(double offset, double lateralOffset = 0) const {
562 1586170640 : return myShape.positionAtOffset(interpolateLanePosToGeometryPos(offset), lateralOffset);
563 : }
564 :
565 : /* @brief fit the given geometry position to a valid lane position
566 : * (lane length might differ from geometry length) */
567 : inline double interpolateGeometryPosToLanePos(double geometryPos) const {
568 6985276 : return geometryPos / myLengthGeometryFactor;
569 : }
570 :
571 : /** @brief Returns the lane's maximum speed, given a vehicle's speed limit adaptation
572 : * @param[in] The vehicle to return the adapted speed limit for
573 : * @return This lane's resulting max. speed
574 : */
575 4012570677 : inline double getVehicleMaxSpeed(const SUMOTrafficObject* const veh) const {
576 4012570677 : return getVehicleMaxSpeed(veh, veh->getMaxSpeed());
577 : }
578 :
579 :
580 6005748096 : inline double getVehicleMaxSpeed(const SUMOTrafficObject* const veh, double vehMaxSpeed) const {
581 6005748096 : if (myRestrictions != nullptr) {
582 5036756 : std::map<SUMOVehicleClass, double>::const_iterator r = myRestrictions->find(veh->getVClass());
583 5036756 : if (r != myRestrictions->end()) {
584 4842011 : if (mySpeedModified) {
585 32670 : return MIN2(myMaxSpeed, MIN2(vehMaxSpeed, r->second * veh->getChosenSpeedFactor()));
586 : } else {
587 4825676 : return MIN2(vehMaxSpeed, r->second * veh->getChosenSpeedFactor());
588 : }
589 : }
590 : }
591 6000906085 : return MIN2(vehMaxSpeed, myMaxSpeed * veh->getChosenSpeedFactor());
592 : }
593 :
594 :
595 : inline bool isSpeedModified() const {
596 1680 : return mySpeedModified;
597 : }
598 :
599 :
600 : /** @brief Returns the lane's maximum allowed speed
601 : * @return This lane's maximum allowed speed
602 : */
603 : inline double getSpeedLimit() const {
604 600480341 : return myMaxSpeed;
605 : }
606 :
607 : /** @brief Returns the lane's friction coefficient
608 : * @return This lane's friction coefficient
609 : */
610 0 : inline double getFrictionCoefficient() const {
611 4091 : return myFrictionCoefficient;
612 : }
613 :
614 : /** @brief Returns the lane's length
615 : * @return This lane's length
616 : */
617 : inline double getLength() const {
618 38644731552 : return myLength;
619 : }
620 :
621 :
622 : /** @brief Returns the vehicle class permissions for this lane
623 : * @return This lane's allowed vehicle classes
624 : */
625 : inline SVCPermissions getPermissions() const {
626 92063644 : return myPermissions;
627 : }
628 :
629 : /** @brief Returns the vehicle class permissions for changing to the left neighbour lane
630 : * @return The vehicle classes allowed to change to the left neighbour lane
631 : */
632 : inline SVCPermissions getChangeLeft() const {
633 26 : return myChangeLeft;
634 : }
635 :
636 : /** @brief Returns the vehicle class permissions for changing to the right neighbour lane
637 : * @return The vehicle classes allowed to change to the right neighbour lane
638 : */
639 : inline SVCPermissions getChangeRight() const {
640 18 : return myChangeRight;
641 : }
642 :
643 : /** @brief Returns the lane's width
644 : * @return This lane's width
645 : */
646 : double getWidth() const {
647 4903857899 : return myWidth;
648 : }
649 :
650 : /** @brief Returns the lane's index
651 : * @return This lane's index
652 : */
653 : int getIndex() const {
654 6064641211 : return myIndex;
655 : }
656 : /// @}
657 :
658 : /// @brief return the index of the link to the next crossing if this is walkingArea, else -1
659 : int getCrossingIndex() const;
660 :
661 :
662 : /// @name Vehicle movement (longitudinal)
663 : /// @{
664 :
665 : /** @brief Compute safe velocities for all vehicles based on positions and
666 : * speeds from the last time step. Also registers
667 : * ApproachingVehicleInformation for all links
668 : *
669 : * This method goes through all vehicles calling their "planMove" method.
670 : * @see MSVehicle::planMove
671 : */
672 : virtual void planMovements(const SUMOTime t);
673 :
674 : /** @brief Register junction approaches for all vehicles after velocities
675 : * have been planned.
676 : *
677 : * This method goes through all vehicles calling their * "setApproachingForAllLinks" method.
678 : */
679 : virtual void setJunctionApproaches() const;
680 :
681 : /** @brief This updates the MSLeaderInfo argument with respect to the given MSVehicle.
682 : * All leader-vehicles on the same edge, which are relevant for the vehicle
683 : * (i.e. with position > vehicle's position) and not already integrated into
684 : * the LeaderInfo, are integrated.
685 : * The given iterators vehPart and vehRes give access to these vehicles which are
686 : * either partial occupators or have issued a maneuver reservation for the lane
687 : * (the latter occurs only for the sublane model).
688 : */
689 : void updateLeaderInfo(const MSVehicle* veh, VehCont::reverse_iterator& vehPart, VehCont::reverse_iterator& vehRes, MSLeaderInfo& ahead) const;
690 :
691 : /** @brief Executes planned vehicle movements with regards to right-of-way
692 : *
693 : * This method goes through all vehicles calling their executeMove method
694 : * which causes vehicles to update their positions and speeds.
695 : * Vehicles wich move to the next lane are stored in the targets lane buffer
696 : *
697 : * @return Returns true, if all vehicles left the lane.
698 : *
699 : * @see MSVehicle::executeMove
700 : */
701 : virtual void executeMovements(const SUMOTime t);
702 :
703 : /// Insert buffered vehicle into the real lane.
704 : virtual void integrateNewVehicles();
705 :
706 : /** @brief Set a flag to recalculate the brutto (including minGaps) occupancy of this lane (used if mingap is changed)
707 : */
708 : void markRecalculateBruttoSum();
709 :
710 : /// @brief updated current vehicle length sum (delayed to avoid lane-order-dependency)
711 : void updateLengthSum();
712 : ///@}
713 :
714 :
715 : /// @brief short-circut collision check if nothing changed since the last check
716 : inline bool needsCollisionCheck() const {
717 399332584 : return myNeedsCollisionCheck;
718 : }
719 :
720 : /// @brief require another collision check due to relevant changes in the simulation
721 : inline void requireCollisionCheck() {
722 4610986 : myNeedsCollisionCheck = true;
723 1010087 : }
724 :
725 : /// Check if vehicles are too close.
726 : virtual void detectCollisions(SUMOTime timestep, const std::string& stage);
727 :
728 :
729 : /** Returns the information whether this lane may be used to continue
730 : the current route */
731 : virtual bool appropriate(const MSVehicle* veh) const;
732 :
733 :
734 : /// returns the container with all links !!!
735 : const std::vector<MSLink*>& getLinkCont() const {
736 1730 : return myLinks;
737 : }
738 :
739 : /// returns the link to the given lane or nullptr, if it is not connected
740 : const MSLink* getLinkTo(const MSLane* const) const;
741 :
742 : /// returns the internal lane leading to the given lane or nullptr, if there is none
743 : const MSLane* getInternalFollowingLane(const MSLane* const) const;
744 :
745 : /// Returns the entry link if this is an internal lane, else nullptr
746 : const MSLink* getEntryLink() const;
747 :
748 :
749 : /// Returns true if there is not a single vehicle on the lane.
750 : bool empty() const {
751 : assert(myVehBuffer.size() == 0);
752 : return myVehicles.empty();
753 : }
754 :
755 : /** @brief Sets a new maximum speed for the lane (used by TraCI, MSLaneSpeedTrigger (VSS) and MSCalibrator)
756 : * @param[in] val the new speed in m/s
757 : * @param[in] modified whether this modifies the original speed
758 : * @param[in] jamThreshold also set a new jamThreshold
759 : */
760 : void setMaxSpeed(const double val, const bool modified = true, const double jamThreshold = -1);
761 :
762 : /** @brief Sets a new friction coefficient for the lane [*to be later (used by TraCI and MSCalibrator)*]
763 : * @param[in] val the new friction coefficient [0..1]
764 : */
765 : void setFrictionCoefficient(double val);
766 :
767 : /** @brief Sets a new length for the lane (used by TraCI only)
768 : * @param[in] val the new length in m
769 : */
770 : void setLength(double val);
771 :
772 : /** @brief Returns the lane's edge
773 : * @return This lane's edge
774 : */
775 : MSEdge& getEdge() const {
776 50483865296 : return *myEdge;
777 : }
778 :
779 : const MSJunction* getFromJunction() const;
780 : const MSJunction* getToJunction() const;
781 :
782 : /** @brief Returns the lane's follower if it is an internal lane, the edge of the lane otherwise
783 : * @return This lane's follower
784 : */
785 : const MSEdge* getNextNormal() const;
786 :
787 :
788 : /** @brief Returns 0 if the lane is not internal. Otherwise the first part of the
789 : * connection (sequence of internal lanes along junction) corresponding to the lane
790 : * is returned and the offset is set to the distance of the begin of this lane
791 : * to the begin of the returned.
792 : */
793 : const MSLane* getFirstInternalInConnection(double& offset) const;
794 :
795 :
796 : /// @brief Static (sic!) container methods
797 : /// {
798 :
799 : /** @brief Inserts a MSLane into the static dictionary
800 : *
801 : * Returns true if the key id isn't already in the dictionary.
802 : * Otherwise returns false.
803 : * @param[in] id The id of the lane
804 : * @param[in] lane The lane itself
805 : * @return Whether the lane was added
806 : * @todo make non-static
807 : * @todo why is the id given? The lane is named
808 : */
809 : static bool dictionary(const std::string& id, MSLane* lane);
810 :
811 :
812 : /** @brief Returns the MSLane associated to the key id
813 : *
814 : * The lane is returned if exists, otherwise 0 is returned.
815 : * @param[in] id The id of the lane
816 : * @return The lane
817 : */
818 : static MSLane* dictionary(const std::string& id);
819 :
820 :
821 : /** @brief Clears the dictionary */
822 : static void clear();
823 :
824 :
825 : /** @brief Returns the number of stored lanes
826 : * @return The number of stored lanes
827 : */
828 : static int dictSize() {
829 40893 : return (int)myDict.size();
830 : }
831 :
832 :
833 : /** @brief Adds the ids of all stored lanes into the given vector
834 : * @param[in, filled] into The vector to add the IDs into
835 : */
836 : static void insertIDs(std::vector<std::string>& into);
837 :
838 :
839 : /** @brief Fills the given RTree with lane instances
840 : * @param[in, filled] into The RTree to fill
841 : * @see TraCILaneRTree
842 : */
843 : template<class RTREE>
844 : static void fill(RTREE& into);
845 :
846 :
847 : /// @brief initialize rngs
848 : static void initRNGs(const OptionsCont& oc);
849 : /// @}
850 :
851 :
852 :
853 : // XXX: succLink does not exist... Documentation?
854 : /** Same as succLink, but does not throw any assertions when
855 : the succeeding link could not be found;
856 : Returns the myLinks.end() instead; Further, the number of edges to
857 : look forward may be given */
858 : static std::vector<MSLink*>::const_iterator succLinkSec(const SUMOVehicle& veh,
859 : int nRouteSuccs,
860 : const MSLane& succLinkSource,
861 : const std::vector<MSLane*>& conts);
862 :
863 :
864 : /** Returns the information whether the given link shows at the end
865 : of the list of links (is not valid) */
866 : inline bool isLinkEnd(std::vector<MSLink*>::const_iterator& i) const {
867 : return i == myLinks.end();
868 : }
869 :
870 : /** Returns the information whether the given link shows at the end
871 : of the list of links (is not valid) */
872 : inline bool isLinkEnd(std::vector<MSLink*>::iterator& i) {
873 : return i == myLinks.end();
874 : }
875 :
876 : /** Returns the information whether the lane is has no vehicle and no
877 : partial occupation*/
878 : inline bool isEmpty() const {
879 198888 : return myVehicles.empty() && myPartialVehicles.empty();
880 : }
881 :
882 : /** Returns whether the lane pertains to an internal edge*/
883 : bool isInternal() const;
884 :
885 : /** Returns whether the lane pertains to a normal edge*/
886 : bool isNormal() const;
887 :
888 : /** Returns whether the lane pertains to a crossing edge*/
889 : bool isCrossing() const;
890 :
891 : /** Returns whether the lane pertains to a crossing edge*/
892 : bool isPriorityCrossing() const;
893 :
894 : /** Returns whether the lane pertains to a walkingarea*/
895 : bool isWalkingArea() const;
896 :
897 : /// @brief returns the last vehicle for which this lane is responsible or 0
898 : MSVehicle* getLastFullVehicle() const;
899 :
900 : /// @brief returns the first vehicle for which this lane is responsible or 0
901 : MSVehicle* getFirstFullVehicle() const;
902 :
903 : /// @brief returns the last vehicle that is fully or partially on this lane
904 : MSVehicle* getLastAnyVehicle() const;
905 :
906 : /// @brief returns the first vehicle that is fully or partially on this lane
907 : MSVehicle* getFirstAnyVehicle() const;
908 :
909 : /* @brief remove the vehicle from this lane
910 : * @param[notify] whether moveReminders of the vehicle shall be triggered
911 : */
912 : virtual MSVehicle* removeVehicle(MSVehicle* remVehicle, MSMoveReminder::Notification notification, bool notify = true);
913 :
914 : void leftByLaneChange(MSVehicle* v);
915 : void enteredByLaneChange(MSVehicle* v);
916 :
917 : /** @brief Returns the lane with the given offset parallel to this one or 0 if it does not exist
918 : * @param[in] offset The offset of the result lane
919 : */
920 : MSLane* getParallelLane(int offset, bool includeOpposite = true) const;
921 :
922 :
923 : /** @brief Sets the permissions to the given value. If a transientID is given, the permissions are recored as temporary
924 : * @param[in] permissions The new permissions
925 : * @param[in] transientID The id of the permission-modification or the special value PERMANENT
926 : */
927 : void setPermissions(SVCPermissions permissions, long long transientID);
928 : void resetPermissions(long long transientID);
929 : bool hadPermissionChanges() const;
930 :
931 : /** @brief Sets the permissions for changing to the left neighbour lane
932 : * @param[in] permissions The new permissions
933 : */
934 : void setChangeLeft(SVCPermissions permissions);
935 :
936 : /** @brief Sets the permissions for changing to the right neighbour lane
937 : * @param[in] permissions The new permissions
938 : */
939 : void setChangeRight(SVCPermissions permissions);
940 :
941 : inline bool allowsVehicleClass(SUMOVehicleClass vclass) const {
942 3466620072 : return (myPermissions & vclass) == vclass;
943 : }
944 :
945 : bool allowsVehicleClass(SUMOVehicleClass vclass, int routingMode) const;
946 :
947 : /** @brief Returns whether the given vehicle class may change left from this lane */
948 : inline bool allowsChangingLeft(SUMOVehicleClass vclass) const {
949 300465623 : return (myChangeLeft & vclass) == vclass;
950 : }
951 :
952 : /** @brief Returns whether the given vehicle class may change left from this lane */
953 : inline bool allowsChangingRight(SUMOVehicleClass vclass) const {
954 265206779 : return (myChangeRight & vclass) == vclass;
955 : }
956 :
957 : void addIncomingLane(MSLane* lane, MSLink* viaLink);
958 :
959 :
960 : struct IncomingLaneInfo {
961 : MSLane* lane;
962 : double length;
963 : MSLink* viaLink;
964 : };
965 :
966 : const std::vector<IncomingLaneInfo>& getIncomingLanes() const {
967 1460698 : return myIncomingLanes;
968 : }
969 :
970 :
971 : void addApproachingLane(MSLane* lane, bool warnMultiCon);
972 : inline bool isApproachedFrom(MSEdge* const edge) {
973 : return myApproachingLanes.find(edge) != myApproachingLanes.end();
974 : }
975 : bool isApproachedFrom(MSEdge* const edge, MSLane* const lane);
976 :
977 : /// @brief Returns vehicle class specific stopOffset for the vehicle
978 : double getVehicleStopOffset(const MSVehicle* veh) const;
979 :
980 : /// @brief Returns vehicle class specific stopOffsets
981 : const StopOffset& getLaneStopOffsets() const;
982 :
983 : /// @brief Set vehicle class specific stopOffsets
984 : void setLaneStopOffset(const StopOffset& stopOffset);
985 :
986 : /** @enum MinorLinkMode
987 : * @brief determine whether/how getFollowers looks upstream beyond minor links
988 : */
989 : enum MinorLinkMode {
990 : FOLLOW_NEVER = 0,
991 : FOLLOW_ALWAYS = 1,
992 : FOLLOW_ONCOMING = 2,
993 : };
994 :
995 : /// @brief return the sublane followers with the largest missing rear gap among all predecessor lanes (within dist)
996 : MSLeaderDistanceInfo getFollowersOnConsecutive(const MSVehicle* ego, double backOffset,
997 : bool allSublanes, double searchDist = -1, MinorLinkMode mLinkMode = FOLLOW_ALWAYS) const;
998 :
999 : /// @brief return by how much further the leader must be inserted to avoid rear end collisions
1000 : double getMissingRearGap(const MSVehicle* leader, double backOffset, double leaderSpeed) const;
1001 :
1002 : /** @brief Returns the immediate leader of veh and the distance to veh
1003 : * starting on this lane
1004 : *
1005 : * Iterates over the current lane to find a leader and then uses
1006 : * getLeaderOnConsecutive()
1007 : * @param[in] veh The vehicle for which the information shall be computed
1008 : * @param[in] vehPos The vehicle position relative to this lane (may be negative)
1009 : * @param[in] bestLaneConts The succeding lanes that shall be checked (if any)
1010 : * @param[in] dist Optional distance to override default (ego stopDist)
1011 : * @param[in] checkTmpVehicles Whether myTmpVehicles should be used instead of myVehicles
1012 : * @return
1013 : */
1014 : std::pair<MSVehicle* const, double> getLeader(const MSVehicle* veh, const double vehPos, const std::vector<MSLane*>& bestLaneConts, double dist = -1, bool checkTmpVehicles = false) const;
1015 :
1016 : /** @brief Returns the immediate leader and the distance to him
1017 : *
1018 : * Goes along the vehicle's estimated used lanes (bestLaneConts). For each link,
1019 : * it is determined whether the vehicle will pass it. If so, the subsequent lane
1020 : * is investigated. If a vehicle (leader) is found, it is returned, together with the length
1021 : * of the investigated lanes until this vehicle's end, including the already seen
1022 : * place (seen).
1023 : *
1024 : * If no leading vehicle was found, <0, -1> is returned.
1025 : *
1026 : * Pretty slow, as it has to go along lanes.
1027 : *
1028 : * @todo: There are some oddities:
1029 : * - what about crossing a link at red, or if a link is closed? Has a following vehicle to be regarded or not?
1030 : *
1031 : * @param[in] dist The distance to investigate
1032 : * @param[in] seen The already seen place (normally the place in front on own lane)
1033 : * @param[in] speed The speed of the vehicle used for determining whether a subsequent link will be opened at arrival time
1034 : * @param[in] veh The vehicle for which the information shall be computed
1035 : * @param[in] bestLaneConts The lanes the vehicle will use in future
1036 : * @param[in] considerCrossingFoes Whether vehicles on crossing foe links should be considered
1037 : * @return
1038 : */
1039 : std::pair<MSVehicle* const, double> getLeaderOnConsecutive(double dist, double seen,
1040 : double speed, const MSVehicle& veh, const std::vector<MSLane*>& bestLaneConts, bool considerCrossingFoes = true) const;
1041 :
1042 : /// @brief Returns the immediate leaders and the distance to them (as getLeaderOnConsecutive but for the sublane case)
1043 : void getLeadersOnConsecutive(double dist, double seen, double speed, const MSVehicle* ego,
1044 : const std::vector<MSLane*>& bestLaneConts, MSLeaderDistanceInfo& result, bool oppositeDirection = false) const;
1045 :
1046 :
1047 : /// @brief get leaders for ego on the given lane
1048 : void addLeaders(const MSVehicle* vehicle, double vehPos, MSLeaderDistanceInfo& result, bool oppositeDirection = false);
1049 :
1050 :
1051 : /** @brief Returns the most dangerous leader and the distance to him
1052 : *
1053 : * Goes along the vehicle's estimated used lanes (bestLaneConts). For each link,
1054 : * it is determined whether the ego vehicle will pass it. If so, the subsequent lane
1055 : * is investigated. Check all lanes up to the stopping distance of ego.
1056 : * Return the leader vehicle (and the gap) which puts the biggest speed constraint on ego.
1057 : *
1058 : * If no leading vehicle was found, <0, -1> is returned.
1059 : *
1060 : * Pretty slow, as it has to go along lanes.
1061 : *
1062 : * @param[in] dist The distance to investigate
1063 : * @param[in] seen The already seen place (normally the place in front on own lane)
1064 : * @param[in] speed The speed of the vehicle used for determining whether a subsequent link will be opened at arrival time
1065 : * @param[in] veh The (ego) vehicle for which the information shall be computed
1066 : * @return
1067 : */
1068 : std::pair<MSVehicle* const, double> getCriticalLeader(double dist, double seen, double speed, const MSVehicle& veh) const;
1069 :
1070 : /* @brief return the partial vehicle closest behind ego or 0
1071 : * if no such vehicle exists */
1072 : MSVehicle* getPartialBehind(const MSVehicle* ego) const;
1073 :
1074 : /// @brief get all vehicles that are inlapping from consecutive edges
1075 : MSLeaderInfo getPartialBeyond() const;
1076 :
1077 : /// @brief Returns all vehicles closer than downstreamDist along the road network starting on the given
1078 : /// position. Predecessor lanes are searched upstream for the given upstreamDistance.
1079 : /// @note Re-implementation of the corresponding method in MSDevice_SSM, which cannot be easily adapted, as it gathers
1080 : /// additional information for conflict lanes, etc.
1081 : /// @param[in] startPos - start position of the search on the first lane
1082 : /// @param[in] downstreamDist - distance to search downstream
1083 : /// @param[in] upstreamDist - distance to search upstream
1084 : /// @param[in/out] checkedLanes - lanes, which were already scanned (current lane is added, if not present,
1085 : /// otherwise the scan is aborted; TODO: this may disregard unscanned parts of the lane in specific circular set ups.)
1086 : /// @return vehs - List of vehicles found
1087 : std::set<MSVehicle*> getSurroundingVehicles(double startPos, double downstreamDist, double upstreamDist, std::shared_ptr<LaneCoverageInfo> checkedLanes) const;
1088 :
1089 : /// @brief Returns all vehicles on the lane overlapping with the interval [a,b]
1090 : /// @note Does not consider vehs with front on subsequent lanes
1091 : std::set<MSVehicle*> getVehiclesInRange(const double a, const double b) const;
1092 :
1093 : /// @brief Returns all upcoming junctions within given range along the given (non-internal) continuation lanes measured from given position
1094 : std::vector<const MSJunction*> getUpcomingJunctions(double pos, double range, const std::vector<MSLane*>& contLanes) const;
1095 :
1096 : /// @brief Returns all upcoming links within given range along the given (non-internal) continuation lanes measured from given position
1097 : std::vector<const MSLink*> getUpcomingLinks(double pos, double range, const std::vector<MSLane*>& contLanes) const;
1098 :
1099 : /** @brief get the most likely precedecessor lane (sorted using by_connections_to_sorter).
1100 : * The result is cached in myLogicalPredecessorLane
1101 : */
1102 : MSLane* getLogicalPredecessorLane() const;
1103 :
1104 : /** @brief get normal lane leading to this internal lane, for normal lanes,
1105 : * the lane itself is returned
1106 : */
1107 : const MSLane* getNormalPredecessorLane() const;
1108 :
1109 : /** @brief get normal lane following this internal lane, for normal lanes,
1110 : * the lane itself is returned
1111 : */
1112 : const MSLane* getNormalSuccessorLane() const;
1113 :
1114 : /** @brief return the (first) predecessor lane from the given edge
1115 : */
1116 : MSLane* getLogicalPredecessorLane(const MSEdge& fromEdge) const;
1117 :
1118 :
1119 : /** Return the main predecessor lane for the current.
1120 : * If there are several incoming lanes, the first attempt is to return the priorized.
1121 : * If this does not yield an unambiguous lane, the one with the least angle difference
1122 : * to the current is selected.
1123 : */
1124 : MSLane* getCanonicalPredecessorLane() const;
1125 :
1126 :
1127 : /** Return the main successor lane for the current.
1128 : * If there are several outgoing lanes, the first attempt is to return the priorized.
1129 : * If this does not yield an unambiguous lane, the one with the least angle difference
1130 : * to the current is selected.
1131 : */
1132 : MSLane* getCanonicalSuccessorLane() const;
1133 :
1134 : /// @brief get the state of the link from the logical predecessor to this lane
1135 : LinkState getIncomingLinkState() const;
1136 :
1137 : /// @brief get the list of outgoing lanes
1138 : const std::vector<std::pair<const MSLane*, const MSEdge*> > getOutgoingViaLanes() const;
1139 :
1140 : /// @brief get the list of all direct (disregarding internal predecessors) non-internal predecessor lanes of this lane
1141 : std::vector<const MSLane*> getNormalIncomingLanes() const;
1142 :
1143 : /// @name Current state retrieval
1144 : //@{
1145 :
1146 : /** @brief Returns the mean speed on this lane
1147 : * @return The average speed of vehicles during the last step; default speed if no vehicle was on this lane
1148 : */
1149 : double getMeanSpeed() const;
1150 :
1151 : /// @brief get the mean speed of all bicycles on this lane
1152 : double getMeanSpeedBike() const;
1153 :
1154 : /** @brief Returns the overall waiting time on this lane
1155 : * @return The sum of the waiting time of all vehicles during the last step;
1156 : */
1157 : double getWaitingSeconds() const;
1158 :
1159 :
1160 : /** @brief Returns the brutto (including minGaps) occupancy of this lane during the last step
1161 : * @return The occupancy during the last step
1162 : */
1163 : double getBruttoOccupancy() const;
1164 :
1165 :
1166 : /** @brief Returns the netto (excluding minGaps) occupancy of this lane during the last step (including minGaps)
1167 : * @return The occupancy during the last step
1168 : */
1169 : double getNettoOccupancy() const;
1170 :
1171 :
1172 : /** @brief Returns the sum of lengths of vehicles, including their minGaps, which were on the lane during the last step
1173 : * @return The sum of vehicle lengths of vehicles in the last step
1174 : */
1175 : inline double getBruttoVehLenSum() const {
1176 2753550271 : return myBruttoVehicleLengthSum;
1177 : }
1178 :
1179 :
1180 : /** @brief Returns the sum of last step emissions
1181 : * The value is always per 1s, so multiply by step length if necessary.
1182 : * @return emissions of vehicles on this lane during the last step
1183 : */
1184 : template<PollutantsInterface::EmissionType ET>
1185 327921 : double getEmissions() const {
1186 : double ret = 0;
1187 332957 : for (MSVehicle* const v : getVehiclesSecure()) {
1188 5036 : ret += v->getEmissions<ET>();
1189 : }
1190 327921 : releaseVehicles();
1191 327921 : return ret;
1192 : }
1193 :
1194 :
1195 : /** @brief Returns the sum of last step noise emissions
1196 : * @return noise emissions of vehicles on this lane during the last step
1197 : */
1198 : double getHarmonoise_NoiseEmissions() const;
1199 : /// @}
1200 :
1201 : void setRightSideOnEdge(double value, int rightmostSublane) {
1202 2059674 : myRightSideOnEdge = value;
1203 2059674 : myRightmostSublane = rightmostSublane;
1204 : }
1205 :
1206 : /// @brief initialized vClass-specific speed limits
1207 : void initRestrictions();
1208 :
1209 : void checkBufferType();
1210 :
1211 : double getRightSideOnEdge() const {
1212 5198362437 : return myRightSideOnEdge;
1213 : }
1214 :
1215 : int getRightmostSublane() const {
1216 101302044 : return myRightmostSublane;
1217 : }
1218 :
1219 : double getCenterOnEdge() const {
1220 26436 : return myRightSideOnEdge + 0.5 * myWidth;
1221 : }
1222 :
1223 : /// @brief sorts myPartialVehicles
1224 : void sortPartialVehicles();
1225 :
1226 : /// @brief sorts myManeuverReservations
1227 : void sortManeuverReservations();
1228 :
1229 : /// @brief return the neighboring opposite direction lane for lane changing or nullptr
1230 : MSLane* getOpposite() const;
1231 :
1232 : /// @brief return the opposite direction lane of this lanes edge or nullptr
1233 : MSLane* getParallelOpposite() const;
1234 :
1235 : /// @brief return the corresponding position on the opposite lane
1236 : double getOppositePos(double pos) const;
1237 :
1238 : /* @brief find leader for a vehicle depending on the relative driving direction
1239 : * @param[in] ego The ego vehicle
1240 : * @param[in] dist The look-ahead distance when looking at consecutive lanes
1241 : * @param[in] oppositeDir Whether the lane has the opposite driving direction of ego
1242 : * @return the leader vehicle and its gap to ego
1243 : */
1244 : std::pair<MSVehicle* const, double> getOppositeLeader(const MSVehicle* ego, double dist, bool oppositeDir, MinorLinkMode mLinkMode = MinorLinkMode::FOLLOW_NEVER) const;
1245 :
1246 : /* @brief find follower for a vehicle that is located on the opposite of this lane
1247 : * @param[in] ego The ego vehicle
1248 : * @return the follower vehicle and its gap to ego
1249 : */
1250 : std::pair<MSVehicle* const, double> getOppositeFollower(const MSVehicle* ego) const;
1251 :
1252 :
1253 : /** @brief Find follower vehicle for the given ego vehicle (which may be on the opposite direction lane)
1254 : * @param[in] ego The ego vehicle
1255 : * @param[in] egoPos The ego position mapped to the current lane
1256 : * @param[in] dist The look-back distance when looking at consecutive lanes
1257 : * @param[in] ignoreMinorLinks Whether backward search should stop at minor links
1258 : * @return the follower vehicle and its gap to ego
1259 : */
1260 : std::pair<MSVehicle* const, double> getFollower(const MSVehicle* ego, double egoPos, double dist, MinorLinkMode mLinkMode) const;
1261 :
1262 :
1263 : ///@brief add parking vehicle. This should only used during state loading
1264 : void addParking(MSBaseVehicle* veh);
1265 :
1266 : ///@brief remove parking vehicle. This must be syncrhonized when running with GUI
1267 : virtual void removeParking(MSBaseVehicle* veh);
1268 :
1269 : /// @brief retrieve the parking vehicles (see GUIParkingArea)
1270 : const std::set<const MSBaseVehicle*>& getParkingVehicles() const {
1271 : return myParkingVehicles;
1272 : }
1273 :
1274 : /// @brief whether this lane is selected in the GUI
1275 1092442 : virtual bool isSelected() const {
1276 1092442 : return false;
1277 : }
1278 :
1279 : /// @brief retrieve bidirectional lane or nullptr
1280 : MSLane* getBidiLane() const;
1281 :
1282 : /// @brief whether this lane must check for junction collisions
1283 : bool mustCheckJunctionCollisions() const;
1284 :
1285 : #ifdef HAVE_FOX
1286 : MFXWorkerThread::Task* getPlanMoveTask(const SUMOTime time) {
1287 : mySimulationTask.init(&MSLane::planMovements, time);
1288 17593954 : return &mySimulationTask;
1289 : }
1290 :
1291 : MFXWorkerThread::Task* getExecuteMoveTask(const SUMOTime time) {
1292 : mySimulationTask.init(&MSLane::executeMovements, time);
1293 17593954 : return &mySimulationTask;
1294 : }
1295 :
1296 : MFXWorkerThread::Task* getLaneChangeTask(const SUMOTime time) {
1297 : mySimulationTask.init(&MSLane::changeLanes, time);
1298 : return &mySimulationTask;
1299 : }
1300 : #endif
1301 :
1302 : std::vector<StopWatch<std::chrono::nanoseconds> >& getStopWatch() {
1303 : return myStopWatch;
1304 : }
1305 :
1306 : void changeLanes(const SUMOTime time);
1307 :
1308 : /// @name State saving/loading
1309 : /// @{
1310 :
1311 : /** @brief Saves the state of this lane into the given stream
1312 : *
1313 : * Basically, a list of vehicle ids
1314 : *
1315 : * @param[in, filled] out The (possibly binary) device to write the state into
1316 : * @todo What about throwing an IOError?
1317 : */
1318 : void saveState(OutputDevice& out);
1319 :
1320 : /** @brief Remove all vehicles before quick-loading state */
1321 : void clearState();
1322 :
1323 : /** @brief Loads the state of this segment with the given parameters
1324 : *
1325 : * This method is called for every internal que the segment has.
1326 : * Every vehicle is retrieved from the given MSVehicleControl and added to this
1327 : * lane.
1328 : *
1329 : * @param[in] vehs The vehicles for the current lane
1330 : * @todo What about throwing an IOError?
1331 : * @todo What about throwing an error if something else fails (a vehicle can not be referenced)?
1332 : */
1333 : void loadState(const std::vector<SUMOVehicle*>& vehs);
1334 :
1335 :
1336 : /* @brief helper function for state saving: checks whether any outgoing
1337 : * links are being approached */
1338 : bool hasApproaching() const;
1339 :
1340 : /// @}
1341 :
1342 :
1343 : /** @brief Callback for visiting the lane when traversing an RTree
1344 : *
1345 : * This is used in the TraCIServerAPI_Lane for context subscriptions.
1346 : *
1347 : * @param[in] cont The context doing all the work
1348 : * @see libsumo::Helper::LaneStoringVisitor::add
1349 : */
1350 1694752 : void visit(const MSLane::StoringVisitor& cont) const {
1351 1694752 : cont.add(this);
1352 1694752 : }
1353 :
1354 : /// @brief whether the lane has pedestrians on it
1355 : bool hasPedestrians() const;
1356 :
1357 : /// This is just a wrapper around MSPModel::nextBlocking. You should always check using hasPedestrians before calling this method.
1358 : std::pair<const MSPerson*, double> nextBlocking(double minPos, double minRight, double maxLeft, double stopTime = 0, bool bidi = false) const;
1359 :
1360 : /// @brief return the empty space up to the last standing vehicle or the empty space on the whole lane if no vehicle is standing
1361 : double getSpaceTillLastStanding(const MSVehicle* ego, bool& foundStopped) const;
1362 :
1363 : /// @brief compute maximum braking distance on this lane
1364 : double getMaximumBrakeDist() const;
1365 :
1366 : inline const PositionVector* getOutlineShape() const {
1367 : return myOutlineShape;
1368 : }
1369 :
1370 : static void initCollisionOptions(const OptionsCont& oc);
1371 : static void initCollisionAction(const OptionsCont& oc, const std::string& option, CollisionAction& myAction);
1372 :
1373 : static CollisionAction getCollisionAction() {
1374 6198 : return myCollisionAction;
1375 : }
1376 :
1377 : static CollisionAction getIntermodalCollisionAction() {
1378 : return myIntermodalCollisionAction;
1379 : }
1380 :
1381 : static DepartSpeedDefinition& getDefaultDepartSpeedDefinition() {
1382 : return myDefaultDepartSpeedDefinition;
1383 : }
1384 :
1385 : static double& getDefaultDepartSpeed() {
1386 : return myDefaultDepartSpeed;
1387 : }
1388 :
1389 :
1390 : static const long CHANGE_PERMISSIONS_PERMANENT = 0;
1391 : static const long CHANGE_PERMISSIONS_GUI = 1;
1392 :
1393 : protected:
1394 : /// moves myTmpVehicles int myVehicles after a lane change procedure
1395 : virtual void swapAfterLaneChange(SUMOTime t);
1396 :
1397 : /** @brief Inserts the vehicle into this lane, and informs it about entering the network
1398 : *
1399 : * Calls the vehicles enterLaneAtInsertion function,
1400 : * updates statistics and modifies the active state as needed
1401 : * @param[in] veh The vehicle to be incorporated
1402 : * @param[in] pos The position of the vehicle
1403 : * @param[in] speed The speed of the vehicle
1404 : * @param[in] posLat The lateral position of the vehicle
1405 : * @param[in] at
1406 : * @param[in] notification The cause of insertion (i.e. departure, teleport, parking) defaults to departure
1407 : */
1408 : virtual void incorporateVehicle(MSVehicle* veh, double pos, double speed, double posLat,
1409 : const MSLane::VehCont::iterator& at,
1410 : MSMoveReminder::Notification notification = MSMoveReminder::NOTIFICATION_DEPARTED);
1411 :
1412 : /// @brief detect whether a vehicle collids with pedestrians on the junction
1413 : void detectPedestrianJunctionCollision(const MSVehicle* collider, const PositionVector& colliderBoundary, const MSLane* foeLane,
1414 : SUMOTime timestep, const std::string& stage,
1415 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toRemove,
1416 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toTeleport);
1417 :
1418 : /// @brief detect whether there is a collision between the two vehicles
1419 : bool detectCollisionBetween(SUMOTime timestep, const std::string& stage, MSVehicle* collider, MSVehicle* victim,
1420 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toRemove,
1421 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toTeleport) const;
1422 :
1423 : /// @brief take action upon collision
1424 : void handleCollisionBetween(SUMOTime timestep, const std::string& stage, const MSVehicle* collider, const MSVehicle* victim,
1425 : double gap, double latGap,
1426 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toRemove,
1427 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toTeleport) const;
1428 :
1429 : void handleIntermodalCollisionBetween(SUMOTime timestep, const std::string& stage, const MSVehicle* collider, const MSTransportable* victim,
1430 : double gap, const std::string& collisionType,
1431 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toRemove,
1432 : std::set<const MSVehicle*, ComparatorNumericalIdLess>& toTeleport) const;
1433 :
1434 : /* @brief determine depart speed and whether it may be patched
1435 : * @param[in] veh The departing vehicle
1436 : * @param[out] whether the speed may be patched to account for safety
1437 : * @return the depart speed
1438 : */
1439 : double getDepartSpeed(const MSVehicle& veh, bool& patchSpeed);
1440 :
1441 : /* @brief determine the lateral depart position
1442 : * @param[in] veh The departing vehicle
1443 : * @return the lateral depart position
1444 : */
1445 : double getDepartPosLat(const MSVehicle& veh);
1446 :
1447 : /** @brief return the maximum safe speed for insertion behind leaders
1448 : * (a negative value indicates that safe insertion is impossible) */
1449 : double safeInsertionSpeed(const MSVehicle* veh, double seen, const MSLeaderInfo& leaders, double speed);
1450 :
1451 : /// @brief check whether pedestrians on this lane interfere with vehicle insertion
1452 : bool checkForPedestrians(const MSVehicle* aVehicle, double& speed, double& dist, double pos, bool patchSpeed) const;
1453 :
1454 : /// @brief check whether any of the outgoing links are being approached
1455 : bool hasApproaching(const std::vector<MSLink*>& links) const;
1456 :
1457 : /// @brief return length of fractional vehicles on this lane
1458 : double getFractionalVehicleLength(bool brutto) const;
1459 :
1460 : /// @brief detect frontal collisions
1461 : static bool isFrontalCollision(const MSVehicle* collider, const MSVehicle* victim);
1462 :
1463 : /// Unique numerical ID (set on reading by netload)
1464 : int myNumericalID;
1465 :
1466 : /// The shape of the lane
1467 : PositionVector myShape;
1468 :
1469 : /// @brief the outline of the lane (optional)
1470 : PositionVector* myOutlineShape = nullptr;
1471 :
1472 : /// The lane index
1473 : int myIndex;
1474 :
1475 : /** @brief The lane's vehicles.
1476 : This container holds all vehicles that have their front (longitudinally)
1477 : and their center (laterally) on this lane.
1478 : These are the vehicles that this lane is 'responsibly' for (i.e. when executing movements)
1479 :
1480 : The entering vehicles are inserted at the front
1481 : of this container and the leaving ones leave from the back, e.g. the
1482 : vehicle in front of the junction (often called first) is
1483 : myVehicles.back() (if it exists). And if it is an iterator at a
1484 : vehicle, ++it points to the vehicle in front. This is the interaction
1485 : vehicle. */
1486 : VehCont myVehicles;
1487 :
1488 : /** @brief The lane's partial vehicles.
1489 : This container holds all vehicles that are partially on this lane but which are
1490 : in myVehicles of another lane.
1491 : Reasons for partial occupancies include the following
1492 : - the back is still on this lane during regular movement
1493 : - the vehicle is performing a continuous lane-change maneuver
1494 : - sub-lane simulation where vehicles can freely move laterally among the lanes of an edge
1495 :
1496 : The entering vehicles are inserted at the front
1497 : of this container and the leaving ones leave from the back. */
1498 : VehCont myPartialVehicles;
1499 :
1500 : /** @brief Container for lane-changing vehicles. After completion of lane-change-
1501 : process, the containers will be swapped with myVehicles. */
1502 : VehCont myTmpVehicles;
1503 :
1504 : /** @brief Buffer for vehicles that moved from their previous lane onto this one.
1505 : * Integrated after all vehicles executed their moves*/
1506 : MFXSynchQue<MSVehicle*, std::vector<MSVehicle*> > myVehBuffer;
1507 :
1508 : /** @brief The vehicles which registered maneuvering into the lane within their current action step.
1509 : * This is currently only relevant for sublane simulation, since continuous lanechanging
1510 : * uses the partial vehicle mechanism.
1511 : *
1512 : * The entering vehicles are inserted at the front
1513 : * of this container and the leaving ones leave from the back. */
1514 : VehCont myManeuverReservations;
1515 :
1516 : /* @brief list of vehicles that are parking near this lane
1517 : * (not necessarily on the road but having reached their stop on this lane)
1518 : * */
1519 : std::set<const MSBaseVehicle*> myParkingVehicles;
1520 :
1521 : /// Lane length [m]
1522 : double myLength;
1523 :
1524 : /// Lane width [m]
1525 : const double myWidth;
1526 :
1527 : /// Lane's vClass specific stop offset [m]. The map is either of length 0, which means no
1528 : /// special stopOffset was set, or of length 1, where the key is a bitset representing a subset
1529 : /// of the SUMOVehicleClass Enum and the value is the offset in meters.
1530 : StopOffset myLaneStopOffset;
1531 :
1532 : /// The lane's edge, for routing only.
1533 : MSEdge* const myEdge;
1534 :
1535 : /// Lane-wide speed limit [m/s]
1536 : double myMaxSpeed;
1537 : /// Lane-wide friction coefficient [0..1]
1538 : double myFrictionCoefficient;
1539 :
1540 : /// @brief Whether the current speed limit is set by a variable speed sign (VSS), TraCI or a MSCalibrator
1541 : bool mySpeedModified;
1542 :
1543 : /// The vClass permissions for this lane
1544 : SVCPermissions myPermissions;
1545 :
1546 : /// The vClass permissions for changing from this lane
1547 : SVCPermissions myChangeLeft;
1548 : SVCPermissions myChangeRight;
1549 :
1550 : /// The original vClass permissions for this lane (before temporary modifications)
1551 : SVCPermissions myOriginalPermissions;
1552 :
1553 : /// The vClass speed restrictions for this lane
1554 : const std::map<SUMOVehicleClass, double>* myRestrictions;
1555 :
1556 : /// All direct predecessor lanes
1557 : std::vector<IncomingLaneInfo> myIncomingLanes;
1558 :
1559 : ///
1560 : mutable MSLane* myLogicalPredecessorLane;
1561 :
1562 : /// Similar to LogicalPredecessorLane, @see getCanonicalPredecessorLane()
1563 : mutable MSLane* myCanonicalPredecessorLane;
1564 :
1565 : /// Main successor lane, @see getCanonicalSuccessorLane()
1566 : mutable MSLane* myCanonicalSuccessorLane;
1567 :
1568 : /// @brief The current length of all vehicles on this lane, including their minGaps
1569 : double myBruttoVehicleLengthSum;
1570 :
1571 : /// @brief The current length of all vehicles on this lane, excluding their minGaps
1572 : double myNettoVehicleLengthSum;
1573 :
1574 : /// @brief The length of all vehicles that have left this lane in the current step (this lane, including their minGaps)
1575 : double myBruttoVehicleLengthSumToRemove;
1576 :
1577 : /// @brief The length of all vehicles that have left this lane in the current step (this lane, excluding their minGaps)
1578 : double myNettoVehicleLengthSumToRemove;
1579 :
1580 : /// @brief Flag to recalculate the occupancy (including minGaps) after a change in minGap
1581 : bool myRecalculateBruttoSum;
1582 :
1583 : /** The lane's Links to its succeeding lanes and the default
1584 : right-of-way rule, i.e. blocked or not blocked. */
1585 : std::vector<MSLink*> myLinks;
1586 :
1587 : /// All direct internal and direct (disregarding internal predecessors) non-internal predecessor lanes of this lane
1588 : std::map<MSEdge*, std::vector<MSLane*> > myApproachingLanes;
1589 :
1590 : /// @brief leaders on all sublanes as seen by approaching vehicles (cached)
1591 : mutable MSLeaderInfo myLeaderInfo;
1592 : /// @brief followers on all sublanes as seen by vehicles on consecutive lanes (cached)
1593 : mutable MSLeaderInfo myFollowerInfo;
1594 :
1595 : /// @brief time step for which myLeaderInfo was last updated
1596 : mutable SUMOTime myLeaderInfoTime;
1597 : /// @brief time step for which myFollowerInfo was last updated
1598 : mutable SUMOTime myFollowerInfoTime;
1599 :
1600 : /// @brief precomputed myShape.length / myLength
1601 : const double myLengthGeometryFactor;
1602 :
1603 : /// @brief whether this lane is an acceleration lane
1604 : const bool myIsRampAccel;
1605 :
1606 : /// @brief the type of this lane
1607 : const std::string myLaneType;
1608 :
1609 : /// @brief the combined width of all lanes with lower index on myEdge
1610 : double myRightSideOnEdge;
1611 : /// @brief the index of the rightmost sublane of this lane on myEdge
1612 : int myRightmostSublane;
1613 :
1614 : /// @brief whether a collision check is currently needed
1615 : bool myNeedsCollisionCheck;
1616 :
1617 : // @brief the neighboring opposite direction or nullptr
1618 : MSLane* myOpposite;
1619 :
1620 : // @brief bidi lane or nullptr
1621 : MSLane* myBidiLane;
1622 :
1623 : // @brief transient changes in permissions
1624 : std::map<long long, SVCPermissions> myPermissionChanges;
1625 :
1626 : // @brief index of the associated thread-rng
1627 : int myRNGIndex;
1628 :
1629 : /// definition of the static dictionary type
1630 : typedef std::map< std::string, MSLane* > DictType;
1631 :
1632 : /// Static dictionary to associate string-ids with objects.
1633 : static DictType myDict;
1634 :
1635 : static std::vector<SumoRNG> myRNGs;
1636 :
1637 : private:
1638 : /// @brief This lane's move reminder
1639 : std::vector< MSMoveReminder* > myMoveReminders;
1640 :
1641 : /// @brief the action to take on collisions
1642 : static CollisionAction myCollisionAction;
1643 : static CollisionAction myIntermodalCollisionAction;
1644 : static bool myCheckJunctionCollisions;
1645 : static double myCheckJunctionCollisionMinGap;
1646 : static SUMOTime myCollisionStopTime;
1647 : static SUMOTime myIntermodalCollisionStopTime;
1648 : static double myCollisionMinGapFactor;
1649 : static bool myExtrapolateSubstepDepart;
1650 : static DepartSpeedDefinition myDefaultDepartSpeedDefinition;
1651 : static double myDefaultDepartSpeed;
1652 : /**
1653 : * @class vehicle_position_sorter
1654 : * @brief Sorts vehicles by their position (descending)
1655 : */
1656 : class vehicle_position_sorter {
1657 : public:
1658 : /// @brief Constructor
1659 : explicit vehicle_position_sorter(const MSLane* lane) :
1660 : myLane(lane) {
1661 : }
1662 :
1663 :
1664 : /** @brief Comparing operator
1665 : * @param[in] v1 First vehicle to compare
1666 : * @param[in] v2 Second vehicle to compare
1667 : * @return Whether the first vehicle is further on the lane than the second
1668 : */
1669 : int operator()(MSVehicle* v1, MSVehicle* v2) const;
1670 :
1671 : const MSLane* myLane;
1672 :
1673 : };
1674 :
1675 : /**
1676 : * @class vehicle_reverse_position_sorter
1677 : * @brief Sorts vehicles by their position (ascending)
1678 : */
1679 : class vehicle_natural_position_sorter {
1680 : public:
1681 : /// @brief Constructor
1682 : explicit vehicle_natural_position_sorter(const MSLane* lane) :
1683 : myLane(lane) {
1684 : }
1685 :
1686 :
1687 : /** @brief Comparing operator
1688 : * @param[in] v1 First vehicle to compare
1689 : * @param[in] v2 Second vehicle to compare
1690 : * @return Whether the first vehicle is further on the lane than the second
1691 : */
1692 : int operator()(MSVehicle* v1, MSVehicle* v2) const;
1693 :
1694 : const MSLane* myLane;
1695 :
1696 : };
1697 :
1698 : /** @class by_connections_to_sorter
1699 : * @brief Sorts edges by their angle relative to the given edge (straight comes first)
1700 : *
1701 : */
1702 : class by_connections_to_sorter {
1703 : public:
1704 : /// @brief constructor
1705 : explicit by_connections_to_sorter(const MSEdge* const e);
1706 :
1707 : /// @brief comparing operator
1708 : int operator()(const MSEdge* const e1, const MSEdge* const e2) const;
1709 :
1710 : private:
1711 : const MSEdge* const myEdge;
1712 : double myLaneDir;
1713 : };
1714 :
1715 :
1716 :
1717 : /** @class incoming_lane_priority_sorter
1718 : * @brief Sorts lanes (IncomingLaneInfos) by their priority or, if this doesn't apply,
1719 : * wrt. the angle difference magnitude relative to the target lane's angle (straight comes first)
1720 : */
1721 : class incoming_lane_priority_sorter {
1722 : public:
1723 : /// @brief constructor
1724 : explicit incoming_lane_priority_sorter(const MSLane* targetLane);
1725 :
1726 : /// @brief comparing operator
1727 : int operator()(const IncomingLaneInfo& lane1, const IncomingLaneInfo& lane2) const;
1728 :
1729 : private:
1730 : const MSLane* const myLane;
1731 : double myLaneDir;
1732 : };
1733 :
1734 :
1735 : /** @class outgoing_lane_priority_sorter
1736 : * @brief Sorts lanes (their origin link) by the priority of their noninternal target edges or, if this doesn't yield an unambiguous result,
1737 : * wrt. the angle difference magnitude relative to the target lane's angle (straight comes first)
1738 : */
1739 : class outgoing_lane_priority_sorter {
1740 : public:
1741 : /// @brief constructor
1742 : explicit outgoing_lane_priority_sorter(const MSLane* sourceLane);
1743 :
1744 : /// @brief comparing operator
1745 : int operator()(const MSLink* link1, const MSLink* link2) const;
1746 :
1747 : private:
1748 : double myLaneDir;
1749 : };
1750 :
1751 : /**
1752 : * @class edge_finder
1753 : */
1754 : class edge_finder {
1755 : public:
1756 6485629 : edge_finder(MSEdge* e) : myEdge(e) {}
1757 : bool operator()(const IncomingLaneInfo& ili) const {
1758 1752897 : return &(ili.lane->getEdge()) == myEdge;
1759 : }
1760 : private:
1761 : const MSEdge* const myEdge;
1762 : };
1763 :
1764 : #ifdef HAVE_FOX
1765 : /// Type of the function that is called for the simulation stage (e.g. planMovements).
1766 : typedef void(MSLane::*Operation)(const SUMOTime);
1767 :
1768 : /**
1769 : * @class SimulationTask
1770 : * @brief the routing task which mainly calls reroute of the vehicle
1771 : */
1772 2041372 : class SimulationTask : public MFXWorkerThread::Task {
1773 : public:
1774 : SimulationTask(MSLane& l, const SUMOTime time)
1775 2059754 : : myLane(l), myTime(time) {}
1776 : void init(Operation operation, const SUMOTime time) {
1777 35187908 : myOperation = operation;
1778 35187908 : myTime = time;
1779 : }
1780 35187908 : void run(MFXWorkerThread* /*context*/) {
1781 : try {
1782 35187908 : (myLane.*(myOperation))(myTime);
1783 4312710 : } catch (ProcessError& e) {
1784 4312710 : WRITE_ERROR(e.what());
1785 4312710 : }
1786 35187908 : }
1787 : private:
1788 : Operation myOperation = nullptr;
1789 : MSLane& myLane;
1790 : SUMOTime myTime;
1791 : private:
1792 : /// @brief Invalidated assignment operator.
1793 : SimulationTask& operator=(const SimulationTask&) = delete;
1794 : };
1795 :
1796 : SimulationTask mySimulationTask;
1797 : /// @brief Mutex for access to the cached leader info value
1798 : mutable FXMutex myLeaderInfoMutex;
1799 : /// @brief Mutex for access to the cached follower info value
1800 : mutable FXMutex myFollowerInfoMutex;
1801 : /// @brief Mutex for access to the cached follower info value
1802 : mutable FXMutex myPartialOccupatorMutex;
1803 : #endif
1804 : std::vector<StopWatch<std::chrono::nanoseconds> > myStopWatch;
1805 :
1806 : private:
1807 : /// @brief invalidated copy constructor
1808 : MSLane(const MSLane&) = delete;
1809 :
1810 : /// @brief invalidated assignment operator
1811 : MSLane& operator=(const MSLane&) = delete;
1812 :
1813 :
1814 : };
1815 :
1816 : // specialized implementation for speedup and avoiding warnings
1817 : #define LANE_RTREE_QUAL RTree<MSLane*, MSLane, float, 2, MSLane::StoringVisitor>
1818 :
1819 : template<>
1820 : inline float LANE_RTREE_QUAL::RectSphericalVolume(Rect* a_rect) {
1821 : ASSERT(a_rect);
1822 633247 : const float extent0 = a_rect->m_max[0] - a_rect->m_min[0];
1823 633247 : const float extent1 = a_rect->m_max[1] - a_rect->m_min[1];
1824 448208 : return .78539816f * (extent0 * extent0 + extent1 * extent1);
1825 : }
1826 :
1827 : template<>
1828 : inline LANE_RTREE_QUAL::Rect LANE_RTREE_QUAL::CombineRect(Rect* a_rectA, Rect* a_rectB) {
1829 : ASSERT(a_rectA && a_rectB);
1830 : Rect newRect;
1831 639734 : newRect.m_min[0] = rtree_min(a_rectA->m_min[0], a_rectB->m_min[0]);
1832 639734 : newRect.m_max[0] = rtree_max(a_rectA->m_max[0], a_rectB->m_max[0]);
1833 639734 : newRect.m_min[1] = rtree_min(a_rectA->m_min[1], a_rectB->m_min[1]);
1834 639734 : newRect.m_max[1] = rtree_max(a_rectA->m_max[1], a_rectB->m_max[1]);
1835 : return newRect;
1836 : }
|