4 * This file is part of OpenTTD.
5 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
6 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
7 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
10 /** @file yapf_costrail.hpp Cost determination for rails. */
12 #ifndef YAPF_COSTRAIL_HPP
13 #define YAPF_COSTRAIL_HPP
15 #include "../../pbs.h"
17 template <class Types
>
18 class CYapfCostRailT
: public CYapfCostBase
{
20 typedef typename
Types::Tpf Tpf
; ///< the pathfinder class (derived from THIS class)
21 typedef typename
Types::TrackFollower TrackFollower
;
22 typedef typename
Types::NodeList::Titem Node
; ///< this will be our node type
23 typedef typename
Node::Key Key
; ///< key to hash tables
24 typedef typename
Node::CachedData CachedData
;
28 /* Structure used inside PfCalcCost() to keep basic tile information. */
38 td
= INVALID_TRACKDIR
;
40 rail_type
= INVALID_RAILTYPE
;
43 TILE(TileIndex tile
, Trackdir td
)
47 this->tile_type
= GetTileType(tile
);
48 this->rail_type
= GetTileRailType(tile
);
55 tile_type
= src
.tile_type
;
56 rail_type
= src
.rail_type
;
62 * @note maximum cost doesn't work with caching enabled
63 * @todo fix maximum cost failing with caching (e.g. FS#2900)
66 CBlobT
<int> m_sig_look_ahead_costs
;
70 bool m_stopped_on_first_two_way_signal
;
73 static const int s_max_segment_cost
= 10000;
75 CYapfCostRailT() : m_max_cost(0), m_disable_cache(false), m_stopped_on_first_two_way_signal(false)
77 /* pre-compute look-ahead penalties into array */
78 int p0
= Yapf().PfGetSettings().rail_look_ahead_signal_p0
;
79 int p1
= Yapf().PfGetSettings().rail_look_ahead_signal_p1
;
80 int p2
= Yapf().PfGetSettings().rail_look_ahead_signal_p2
;
81 int *pen
= m_sig_look_ahead_costs
.GrowSizeNC(Yapf().PfGetSettings().rail_look_ahead_max_signals
);
82 for (uint i
= 0; i
< Yapf().PfGetSettings().rail_look_ahead_max_signals
; i
++) {
83 pen
[i
] = p0
+ i
* (p1
+ i
* p2
);
87 /** to access inherited path finder */
90 return *static_cast<Tpf
*>(this);
94 inline int SlopeCost(TileIndex tile
, Trackdir td
)
96 CPerfStart
perf_cost(Yapf().m_perf_slope_cost
);
97 if (!stSlopeCost(tile
, td
)) return 0;
98 return Yapf().PfGetSettings().rail_slope_penalty
;
101 inline int CurveCost(Trackdir td1
, Trackdir td2
)
103 assert(IsValidTrackdir(td1
));
104 assert(IsValidTrackdir(td2
));
106 if (TrackFollower::Allow90degTurns()
107 && ((TrackdirToTrackdirBits(td2
) & (TrackdirBits
)TrackdirCrossesTrackdirs(td1
)) != 0)) {
108 /* 90-deg curve penalty */
109 cost
+= Yapf().PfGetSettings().rail_curve90_penalty
;
110 } else if (td2
!= NextTrackdir(td1
)) {
111 /* 45-deg curve penalty */
112 cost
+= Yapf().PfGetSettings().rail_curve45_penalty
;
117 inline int SwitchCost(TileIndex tile1
, TileIndex tile2
, DiagDirection exitdir
)
119 if (IsPlainRailTile(tile1
) && IsPlainRailTile(tile2
)) {
120 bool t1
= KillFirstBit(GetTrackBits(tile1
) & DiagdirReachesTracks(ReverseDiagDir(exitdir
))) != TRACK_BIT_NONE
;
121 bool t2
= KillFirstBit(GetTrackBits(tile2
) & DiagdirReachesTracks(exitdir
)) != TRACK_BIT_NONE
;
122 if (t1
&& t2
) return Yapf().PfGetSettings().rail_doubleslip_penalty
;
127 /** Return one tile cost (base cost + level crossing penalty). */
128 inline int OneTileCost(TileIndex
&tile
, Trackdir trackdir
)
132 if (IsDiagonalTrackdir(trackdir
)) {
133 cost
+= YAPF_TILE_LENGTH
;
134 switch (GetTileType(tile
)) {
136 /* Increase the cost for level crossings */
137 if (IsLevelCrossing(tile
)) {
138 cost
+= Yapf().PfGetSettings().rail_crossing_penalty
;
146 /* non-diagonal trackdir */
147 cost
= YAPF_TILE_CORNER_LENGTH
;
152 /** Check for a reserved station platform. */
153 inline bool IsAnyStationTileReserved(TileIndex tile
, Trackdir trackdir
, int skipped
)
155 TileIndexDiff diff
= TileOffsByDiagDir(TrackdirToExitdir(ReverseTrackdir(trackdir
)));
156 for (; skipped
>= 0; skipped
--, tile
+= diff
) {
157 if (HasStationReservation(tile
)) return true;
162 /** The cost for reserved tiles, including skipped ones. */
163 inline int ReservationCost(Node
&n
, TileIndex tile
, Trackdir trackdir
, int skipped
)
165 if (n
.m_num_signals_passed
>= m_sig_look_ahead_costs
.Size() / 2) return 0;
166 if (!IsPbsSignal(n
.m_last_signal_type
)) return 0;
168 if (IsRailStationTile(tile
) && IsAnyStationTileReserved(tile
, trackdir
, skipped
)) {
169 return Yapf().PfGetSettings().rail_pbs_station_penalty
* (skipped
+ 1);
170 } else if (TrackOverlapsTracks(GetReservedTrackbits(tile
), TrackdirToTrack(trackdir
))) {
171 int cost
= Yapf().PfGetSettings().rail_pbs_cross_penalty
;
172 if (!IsDiagonalTrackdir(trackdir
)) cost
= (cost
* YAPF_TILE_CORNER_LENGTH
) / YAPF_TILE_LENGTH
;
173 return cost
* (skipped
+ 1);
178 int SignalCost(Node
&n
, TileIndex tile
, Trackdir trackdir
)
181 /* if there is one-way signal in the opposite direction, then it is not our way */
182 CPerfStart
perf_cost(Yapf().m_perf_other_cost
);
183 if (IsTileType(tile
, MP_RAILWAY
)) {
184 bool has_signal_against
= HasSignalOnTrackdir(tile
, ReverseTrackdir(trackdir
));
185 bool has_signal_along
= HasSignalOnTrackdir(tile
, trackdir
);
186 if (has_signal_against
&& !has_signal_along
&& IsOnewaySignal(tile
, TrackdirToTrack(trackdir
))) {
187 /* one-way signal in opposite direction */
188 n
.m_segment
->m_end_segment_reason
|= ESRB_DEAD_END
;
190 if (has_signal_along
) {
191 SignalState sig_state
= GetSignalStateByTrackdir(tile
, trackdir
);
192 SignalType sig_type
= GetSignalType(tile
, TrackdirToTrack(trackdir
));
194 n
.m_last_signal_type
= sig_type
;
196 /* cache the look-ahead polynomial constant only if we didn't pass more signals than the look-ahead limit is */
197 int look_ahead_cost
= (n
.m_num_signals_passed
< m_sig_look_ahead_costs
.Size()) ? m_sig_look_ahead_costs
.Data()[n
.m_num_signals_passed
] : 0;
198 if (sig_state
!= SIGNAL_STATE_RED
) {
200 n
.flags_u
.flags_s
.m_last_signal_was_red
= false;
201 /* negative look-ahead red-signal penalties would cause problems later, so use them as positive penalties for green signal */
202 if (look_ahead_cost
< 0) {
203 /* add its negation to the cost */
204 cost
-= look_ahead_cost
;
207 /* we have a red signal in our direction
208 * was it first signal which is two-way? */
209 if (!IsPbsSignal(sig_type
) && Yapf().TreatFirstRedTwoWaySignalAsEOL() && n
.flags_u
.flags_s
.m_choice_seen
&& has_signal_against
&& n
.m_num_signals_passed
== 0) {
210 /* yes, the first signal is two-way red signal => DEAD END. Prune this branch... */
211 Yapf().PruneIntermediateNodeBranch();
212 n
.m_segment
->m_end_segment_reason
|= ESRB_DEAD_END
;
213 Yapf().m_stopped_on_first_two_way_signal
= true;
216 n
.m_last_red_signal_type
= sig_type
;
217 n
.flags_u
.flags_s
.m_last_signal_was_red
= true;
219 /* look-ahead signal penalty */
220 if (!IsPbsSignal(sig_type
) && look_ahead_cost
> 0) {
221 /* add the look ahead penalty only if it is positive */
222 cost
+= look_ahead_cost
;
225 /* special signal penalties */
226 if (n
.m_num_signals_passed
== 0) {
229 case SIGTYPE_EXIT
: cost
+= Yapf().PfGetSettings().rail_firstred_exit_penalty
; break; // first signal is red pre-signal-exit
231 case SIGTYPE_ENTRY
: cost
+= Yapf().PfGetSettings().rail_firstred_penalty
; break;
237 n
.m_num_signals_passed
++;
238 n
.m_segment
->m_last_signal_tile
= tile
;
239 n
.m_segment
->m_last_signal_td
= trackdir
;
242 if (has_signal_against
&& IsPbsSignal(GetSignalType(tile
, TrackdirToTrack(trackdir
)))) {
243 cost
+= n
.m_num_signals_passed
< Yapf().PfGetSettings().rail_look_ahead_max_signals
? Yapf().PfGetSettings().rail_pbs_signal_back_penalty
: 0;
250 inline int PlatformLengthPenalty(int platform_length
)
253 const Train
*v
= Yapf().GetVehicle();
255 assert(v
->type
== VEH_TRAIN
);
256 assert(v
->gcache
.cached_total_length
!= 0);
257 int missing_platform_length
= CeilDiv(v
->gcache
.cached_total_length
, TILE_SIZE
) - platform_length
;
258 if (missing_platform_length
< 0) {
259 /* apply penalty for longer platform than needed */
260 cost
+= Yapf().PfGetSettings().rail_longer_platform_penalty
+ Yapf().PfGetSettings().rail_longer_platform_per_tile_penalty
* -missing_platform_length
;
261 } else if (missing_platform_length
> 0) {
262 /* apply penalty for shorter platform than needed */
263 cost
+= Yapf().PfGetSettings().rail_shorter_platform_penalty
+ Yapf().PfGetSettings().rail_shorter_platform_per_tile_penalty
* missing_platform_length
;
269 inline void SetMaxCost(int max_cost
)
271 m_max_cost
= max_cost
;
275 * Called by YAPF to calculate the cost from the origin to the given node.
276 * Calculates only the cost of given node, adds it to the parent node cost
277 * and stores the result into Node::m_cost member
279 inline bool PfCalcCost(Node
&n
, const TrackFollower
*tf
)
281 assert(!n
.flags_u
.flags_s
.m_targed_seen
);
282 assert(tf
->m_new_tile
== n
.m_key
.m_tile
);
283 assert((TrackdirToTrackdirBits(n
.m_key
.m_td
) & tf
->m_new_td_bits
) != TRACKDIR_BIT_NONE
);
285 CPerfStart
perf_cost(Yapf().m_perf_cost
);
287 /* Does the node have some parent node? */
288 bool has_parent
= (n
.m_parent
!= NULL
);
290 /* Do we already have a cached segment? */
291 CachedData
&segment
= *n
.m_segment
;
292 bool is_cached_segment
= (segment
.m_cost
>= 0);
294 int parent_cost
= has_parent
? n
.m_parent
->m_cost
: 0;
296 /* Each node cost contains 2 or 3 main components:
297 * 1. Transition cost - cost of the move from previous node (tile):
298 * - curve cost (or zero for straight move)
301 * - YAPF_TILE_LENGTH for diagonal tiles
302 * - YAPF_TILE_CORNER_LENGTH for non-diagonal tiles
304 * - tile slope penalty (upward slopes)
305 * - red signal penalty
306 * - level crossing penalty
307 * - speed-limit penalty (bridges)
308 * - station platform penalty
309 * - penalty for reversing in the depot
311 * 3. Extra cost (applies to the last node only)
312 * - last red signal penalty
313 * - penalty for too long or too short platform on the destination station
315 int transition_cost
= 0;
318 /* Segment: one or more tiles connected by contiguous tracks of the same type.
319 * Each segment cost includes 'Tile cost' for all its tiles (including the first
320 * and last), and the 'Transition cost' between its tiles. The first transition
321 * cost of segment entry (move from the 'parent' node) is not included!
323 int segment_entry_cost
= 0;
324 int segment_cost
= 0;
326 const Train
*v
= Yapf().GetVehicle();
328 /* start at n.m_key.m_tile / n.m_key.m_td and walk to the end of segment */
329 TILE
cur(n
.m_key
.m_tile
, n
.m_key
.m_td
);
331 /* the previous tile will be needed for transition cost calculations */
332 TILE prev
= !has_parent
? TILE() : TILE(n
.m_parent
->GetLastTile(), n
.m_parent
->GetLastTrackdir());
334 EndSegmentReasonBits end_segment_reason
= ESRB_NONE
;
336 TrackFollower
tf_local(v
, Yapf().GetCompatibleRailTypes(), &Yapf().m_perf_ts_cost
);
339 /* We will jump to the middle of the cost calculator assuming that segment cache is not used. */
340 assert(!is_cached_segment
);
341 /* Skip the first transition cost calculation. */
346 /* Transition cost (cost of the move from previous tile) */
347 transition_cost
= Yapf().CurveCost(prev
.td
, cur
.td
);
348 transition_cost
+= Yapf().SwitchCost(prev
.tile
, cur
.tile
, TrackdirToExitdir(prev
.td
));
350 /* First transition cost counts against segment entry cost, other transitions
351 * inside segment will come to segment cost (and will be cached) */
352 if (segment_cost
== 0) {
353 /* We just entered the loop. First transition cost goes to segment entry cost)*/
354 segment_entry_cost
= transition_cost
;
357 /* It is the right time now to look if we can reuse the cached segment cost. */
358 if (is_cached_segment
) {
359 /* Yes, we already know the segment cost. */
360 segment_cost
= segment
.m_cost
;
361 /* We know also the reason why the segment ends. */
362 end_segment_reason
= segment
.m_end_segment_reason
;
363 /* We will need also some information about the last signal (if it was red). */
364 if (segment
.m_last_signal_tile
!= INVALID_TILE
) {
365 assert(HasSignalOnTrackdir(segment
.m_last_signal_tile
, segment
.m_last_signal_td
));
366 SignalState sig_state
= GetSignalStateByTrackdir(segment
.m_last_signal_tile
, segment
.m_last_signal_td
);
367 bool is_red
= (sig_state
== SIGNAL_STATE_RED
);
368 n
.flags_u
.flags_s
.m_last_signal_was_red
= is_red
;
370 n
.m_last_red_signal_type
= GetSignalType(segment
.m_last_signal_tile
, TrackdirToTrack(segment
.m_last_signal_td
));
373 /* No further calculation needed. */
374 cur
= TILE(n
.GetLastTile(), n
.GetLastTrackdir());
378 /* Other than first transition cost count as the regular segment cost. */
379 segment_cost
+= transition_cost
;
382 no_entry_cost
: // jump here at the beginning if the node has no parent (it is the first node)
384 /* All other tile costs will be calculated here. */
385 segment_cost
+= Yapf().OneTileCost(cur
.tile
, cur
.td
);
387 /* If we skipped some tunnel/bridge/station tiles, add their base cost */
388 segment_cost
+= YAPF_TILE_LENGTH
* tf
->m_tiles_skipped
;
391 segment_cost
+= Yapf().SlopeCost(cur
.tile
, cur
.td
);
393 /* Signal cost (routine can modify segment data). */
394 segment_cost
+= Yapf().SignalCost(n
, cur
.tile
, cur
.td
);
396 /* Reserved tiles. */
397 segment_cost
+= Yapf().ReservationCost(n
, cur
.tile
, cur
.td
, tf
->m_tiles_skipped
);
399 end_segment_reason
= segment
.m_end_segment_reason
;
401 /* Tests for 'potential target' reasons to close the segment. */
402 if (cur
.tile
== prev
.tile
) {
403 /* Penalty for reversing in a depot. */
404 assert(IsRailDepot(cur
.tile
));
405 segment_cost
+= Yapf().PfGetSettings().rail_depot_reverse_penalty
;
407 } else if (IsRailDepotTile(cur
.tile
)) {
408 /* We will end in this pass (depot is possible target) */
409 end_segment_reason
|= ESRB_DEPOT
;
411 } else if (cur
.tile_type
== MP_STATION
&& IsRailWaypoint(cur
.tile
)) {
412 if (v
->current_order
.IsType(OT_GOTO_WAYPOINT
) &&
413 GetStationIndex(cur
.tile
) == v
->current_order
.GetDestination() &&
414 !Waypoint::Get(v
->current_order
.GetDestination())->IsSingleTile()) {
415 /* This waypoint is our destination; maybe this isn't an unreserved
416 * one, so check that and if so see that as the last signal being
417 * red. This way waypoints near stations should work better. */
418 CFollowTrackRail
ft(v
);
419 TileIndex t
= cur
.tile
;
420 Trackdir td
= cur
.td
;
421 /* Arbitrary maximum tiles to follow to avoid infinite loops. */
423 while (ft
.Follow(t
, td
)) {
424 assert(t
!= ft
.m_new_tile
);
426 if (t
== cur
.tile
|| --max_tiles
== 0) {
427 /* We looped back on ourself or found another loop, bail out. */
428 td
= INVALID_TRACKDIR
;
431 if (KillFirstBit(ft
.m_new_td_bits
) != TRACKDIR_BIT_NONE
) {
432 /* We encountered a junction; it's going to be too complex to
433 * handle this perfectly, so just bail out. There is no simple
434 * free path, so try the other possibilities. */
435 td
= INVALID_TRACKDIR
;
438 td
= RemoveFirstTrackdir(&ft
.m_new_td_bits
);
439 /* If this is a safe waiting position we're done searching for it */
440 if (IsSafeWaitingPosition(v
, t
, td
, true, _settings_game
.pf
.forbid_90_deg
)) break;
443 /* In the case this platform is (possibly) occupied we add penalty so the
444 * other platforms of this waypoint are evaluated as well, i.e. we assume
445 * that there is a red signal in the waypoint when it's occupied. */
446 if (td
== INVALID_TRACKDIR
||
447 !IsSafeWaitingPosition(v
, t
, td
, true, _settings_game
.pf
.forbid_90_deg
) ||
448 !IsWaitingPositionFree(v
, t
, td
, _settings_game
.pf
.forbid_90_deg
)) {
449 extra_cost
+= Yapf().PfGetSettings().rail_lastred_penalty
;
452 /* Waypoint is also a good reason to finish. */
453 end_segment_reason
|= ESRB_WAYPOINT
;
455 } else if (tf
->m_is_station
) {
456 /* Station penalties. */
457 uint platform_length
= tf
->m_tiles_skipped
+ 1;
458 /* We don't know yet if the station is our target or not. Act like
459 * if it is pass-through station (not our destination). */
460 segment_cost
+= Yapf().PfGetSettings().rail_station_penalty
* platform_length
;
461 /* We will end in this pass (station is possible target) */
462 end_segment_reason
|= ESRB_STATION
;
464 } else if (TrackFollower::DoTrackMasking() && cur
.tile_type
== MP_RAILWAY
) {
465 /* Searching for a safe tile? */
466 if (HasSignalOnTrackdir(cur
.tile
, cur
.td
) && !IsPbsSignal(GetSignalType(cur
.tile
, TrackdirToTrack(cur
.td
)))) {
467 end_segment_reason
|= ESRB_SAFE_TILE
;
471 /* Apply min/max speed penalties only when inside the look-ahead radius. Otherwise
472 * it would cause desync in MP. */
473 if (n
.m_num_signals_passed
< m_sig_look_ahead_costs
.Size())
476 int max_speed
= tf
->GetSpeedLimit(&min_speed
);
477 int max_veh_speed
= v
->GetDisplayMaxSpeed();
478 if (max_speed
< max_veh_speed
) {
479 extra_cost
+= YAPF_TILE_LENGTH
* (max_veh_speed
- max_speed
) * (4 + tf
->m_tiles_skipped
) / max_veh_speed
;
481 if (min_speed
> max_veh_speed
) {
482 extra_cost
+= YAPF_TILE_LENGTH
* (min_speed
- max_veh_speed
);
486 /* Finish if we already exceeded the maximum path cost (i.e. when
487 * searching for the nearest depot). */
488 if (m_max_cost
> 0 && (parent_cost
+ segment_entry_cost
+ segment_cost
) > m_max_cost
) {
489 end_segment_reason
|= ESRB_PATH_TOO_LONG
;
492 /* Move to the next tile/trackdir. */
494 tf_local
.Init(v
, Yapf().GetCompatibleRailTypes(), &Yapf().m_perf_ts_cost
);
496 if (!tf_local
.Follow(cur
.tile
, cur
.td
)) {
497 assert(tf_local
.m_err
!= TrackFollower::EC_NONE
);
498 /* Can't move to the next tile (EOL?). */
499 if (tf_local
.m_err
== TrackFollower::EC_RAIL_TYPE
) {
500 end_segment_reason
|= ESRB_RAIL_TYPE
;
502 end_segment_reason
|= ESRB_DEAD_END
;
505 if (TrackFollower::DoTrackMasking() && !HasOnewaySignalBlockingTrackdir(cur
.tile
, cur
.td
)) {
506 end_segment_reason
|= ESRB_SAFE_TILE
;
511 /* Check if the next tile is not a choice. */
512 if (KillFirstBit(tf_local
.m_new_td_bits
) != TRACKDIR_BIT_NONE
) {
513 /* More than one segment will follow. Close this one. */
514 end_segment_reason
|= ESRB_CHOICE_FOLLOWS
;
518 /* Gather the next tile/trackdir/tile_type/rail_type. */
519 TILE
next(tf_local
.m_new_tile
, (Trackdir
)FindFirstBit2x64(tf_local
.m_new_td_bits
));
521 if (TrackFollower::DoTrackMasking() && IsTileType(next
.tile
, MP_RAILWAY
)) {
522 if (HasSignalOnTrackdir(next
.tile
, next
.td
) && IsPbsSignal(GetSignalType(next
.tile
, TrackdirToTrack(next
.td
)))) {
523 /* Possible safe tile. */
524 end_segment_reason
|= ESRB_SAFE_TILE
;
525 } else if (HasSignalOnTrackdir(next
.tile
, ReverseTrackdir(next
.td
)) && GetSignalType(next
.tile
, TrackdirToTrack(next
.td
)) == SIGTYPE_PBS_ONEWAY
) {
526 /* Possible safe tile, but not so good as it's the back of a signal... */
527 end_segment_reason
|= ESRB_SAFE_TILE
| ESRB_DEAD_END
;
528 extra_cost
+= Yapf().PfGetSettings().rail_lastred_exit_penalty
;
532 /* Check the next tile for the rail type. */
533 if (next
.rail_type
!= cur
.rail_type
) {
534 /* Segment must consist from the same rail_type tiles. */
535 end_segment_reason
|= ESRB_RAIL_TYPE
;
539 /* Avoid infinite looping. */
540 if (next
.tile
== n
.m_key
.m_tile
&& next
.td
== n
.m_key
.m_td
) {
541 end_segment_reason
|= ESRB_INFINITE_LOOP
;
545 if (segment_cost
> s_max_segment_cost
) {
546 /* Potentially in the infinite loop (or only very long segment?). We should
547 * not force it to finish prematurely unless we are on a regular tile. */
548 if (IsTileType(tf
->m_new_tile
, MP_RAILWAY
)) {
549 end_segment_reason
|= ESRB_SEGMENT_TOO_LONG
;
554 /* Any other reason bit set? */
555 if (end_segment_reason
!= ESRB_NONE
) {
559 /* For the next loop set new prev and cur tile info. */
565 /* Don't consider path any further it if exceeded max_cost. */
566 if (end_segment_reason
& ESRB_PATH_TOO_LONG
) return false;
568 bool target_seen
= false;
569 if ((end_segment_reason
& ESRB_POSSIBLE_TARGET
) != ESRB_NONE
) {
570 /* Depot, station or waypoint. */
571 if (Yapf().PfDetectDestination(cur
.tile
, cur
.td
)) {
572 /* Destination found. */
577 /* Update the segment if needed. */
578 if (!is_cached_segment
) {
579 /* Write back the segment information so it can be reused the next time. */
580 segment
.m_cost
= segment_cost
;
581 segment
.m_end_segment_reason
= end_segment_reason
& ESRB_CACHED_MASK
;
582 /* Save end of segment back to the node. */
583 n
.SetLastTileTrackdir(cur
.tile
, cur
.td
);
586 /* Do we have an excuse why not to continue pathfinding in this direction? */
587 if (!target_seen
&& (end_segment_reason
& ESRB_ABORT_PF_MASK
) != ESRB_NONE
) {
588 /* Reason to not continue. Stop this PF branch. */
592 /* Special costs for the case we have reached our target. */
594 n
.flags_u
.flags_s
.m_targed_seen
= true;
595 /* Last-red and last-red-exit penalties. */
596 if (n
.flags_u
.flags_s
.m_last_signal_was_red
) {
597 if (n
.m_last_red_signal_type
== SIGTYPE_EXIT
) {
598 /* last signal was red pre-signal-exit */
599 extra_cost
+= Yapf().PfGetSettings().rail_lastred_exit_penalty
;
600 } else if (!IsPbsSignal(n
.m_last_red_signal_type
)) {
601 /* Last signal was red, but not exit or path signal. */
602 extra_cost
+= Yapf().PfGetSettings().rail_lastred_penalty
;
606 /* Station platform-length penalty. */
607 if ((end_segment_reason
& ESRB_STATION
) != ESRB_NONE
) {
608 const BaseStation
*st
= BaseStation::GetByTile(n
.GetLastTile());
610 uint platform_length
= st
->GetPlatformLength(n
.GetLastTile(), ReverseDiagDir(TrackdirToExitdir(n
.GetLastTrackdir())));
611 /* Reduce the extra cost caused by passing-station penalty (each station receives it in the segment cost). */
612 extra_cost
-= Yapf().PfGetSettings().rail_station_penalty
* platform_length
;
613 /* Add penalty for the inappropriate platform length. */
614 extra_cost
+= PlatformLengthPenalty(platform_length
);
618 /* total node cost */
619 n
.m_cost
= parent_cost
+ segment_entry_cost
+ segment_cost
+ extra_cost
;
624 inline bool CanUseGlobalCache(Node
&n
) const
626 return !m_disable_cache
627 && (n
.m_parent
!= NULL
)
628 && (n
.m_parent
->m_num_signals_passed
>= m_sig_look_ahead_costs
.Size());
631 inline void ConnectNodeToCachedData(Node
&n
, CachedData
&ci
)
634 if (n
.m_segment
->m_cost
< 0) {
635 n
.m_segment
->m_last_tile
= n
.m_key
.m_tile
;
636 n
.m_segment
->m_last_td
= n
.m_key
.m_td
;
640 void DisableCache(bool disable
)
642 m_disable_cache
= disable
;
646 #endif /* YAPF_COSTRAIL_HPP */