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