OpenTTD Source  20240917-master-g9ab0a47812
yapf_costrail.hpp
Go to the documentation of this file.
1 /*
2  * This file is part of OpenTTD.
3  * 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.
4  * 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.
5  * 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/>.
6  */
7 
10 #ifndef YAPF_COSTRAIL_HPP
11 #define YAPF_COSTRAIL_HPP
12 
13 
14 #include "../../pbs.h"
15 
16 template <class Types>
17 class CYapfCostRailT : public CYapfCostBase {
18 public:
19  typedef typename Types::Tpf Tpf;
20  typedef typename Types::TrackFollower TrackFollower;
21  typedef typename Types::NodeList::Titem Node;
22  typedef typename Node::Key Key;
23  typedef typename Node::CachedData CachedData;
24 
25 protected:
26 
27  /* Structure used inside PfCalcCost() to keep basic tile information. */
28  struct TILE {
29  TileIndex tile;
30  Trackdir td;
31  TileType tile_type;
32  RailType rail_type;
33 
34  TILE()
35  {
36  tile = INVALID_TILE;
37  td = INVALID_TRACKDIR;
38  tile_type = MP_VOID;
39  rail_type = INVALID_RAILTYPE;
40  }
41 
42  TILE(TileIndex tile, Trackdir td)
43  {
44  this->tile = tile;
45  this->td = td;
46  this->tile_type = GetTileType(tile);
47  this->rail_type = GetTileRailType(tile);
48  }
49  };
50 
51 protected:
57  bool m_disable_cache;
58  std::vector<int> m_sig_look_ahead_costs;
59 
60 public:
61  bool m_stopped_on_first_two_way_signal;
62 protected:
63 
64  static const int s_max_segment_cost = 10000;
65 
66  CYapfCostRailT() : m_max_cost(0), m_disable_cache(false), m_stopped_on_first_two_way_signal(false)
67  {
68  /* pre-compute look-ahead penalties into array */
69  int p0 = Yapf().PfGetSettings().rail_look_ahead_signal_p0;
70  int p1 = Yapf().PfGetSettings().rail_look_ahead_signal_p1;
71  int p2 = Yapf().PfGetSettings().rail_look_ahead_signal_p2;
72  m_sig_look_ahead_costs.clear();
73  m_sig_look_ahead_costs.reserve(Yapf().PfGetSettings().rail_look_ahead_max_signals);
74  for (uint i = 0; i < Yapf().PfGetSettings().rail_look_ahead_max_signals; i++) {
75  m_sig_look_ahead_costs.push_back(p0 + i * (p1 + i * p2));
76  }
77  }
78 
80  Tpf &Yapf()
81  {
82  return *static_cast<Tpf *>(this);
83  }
84 
85 public:
86  inline int SlopeCost(TileIndex tile, Trackdir td)
87  {
88  if (!stSlopeCost(tile, td)) return 0;
89  return Yapf().PfGetSettings().rail_slope_penalty;
90  }
91 
92  inline int CurveCost(Trackdir td1, Trackdir td2)
93  {
94  assert(IsValidTrackdir(td1));
95  assert(IsValidTrackdir(td2));
96  int cost = 0;
97  if (TrackFollower::Allow90degTurns()
98  && HasTrackdir(TrackdirCrossesTrackdirs(td1), td2)) {
99  /* 90-deg curve penalty */
100  cost += Yapf().PfGetSettings().rail_curve90_penalty;
101  } else if (td2 != NextTrackdir(td1)) {
102  /* 45-deg curve penalty */
103  cost += Yapf().PfGetSettings().rail_curve45_penalty;
104  }
105  return cost;
106  }
107 
108  inline int SwitchCost(TileIndex tile1, TileIndex tile2, DiagDirection exitdir)
109  {
110  if (IsPlainRailTile(tile1) && IsPlainRailTile(tile2)) {
112  bool t2 = KillFirstBit(GetTrackBits(tile2) & DiagdirReachesTracks(exitdir)) != TRACK_BIT_NONE;
113  if (t1 && t2) return Yapf().PfGetSettings().rail_doubleslip_penalty;
114  }
115  return 0;
116  }
117 
119  inline int OneTileCost(TileIndex &tile, Trackdir trackdir)
120  {
121  int cost = 0;
122  /* set base cost */
123  if (IsDiagonalTrackdir(trackdir)) {
124  cost += YAPF_TILE_LENGTH;
125  switch (GetTileType(tile)) {
126  case MP_ROAD:
127  /* Increase the cost for level crossings */
128  if (IsLevelCrossing(tile)) {
129  cost += Yapf().PfGetSettings().rail_crossing_penalty;
130  }
131  break;
132 
133  default:
134  break;
135  }
136  } else {
137  /* non-diagonal trackdir */
139  }
140  return cost;
141  }
142 
144  inline bool IsAnyStationTileReserved(TileIndex tile, Trackdir trackdir, int skipped)
145  {
147  for (; skipped >= 0; skipped--, tile += diff) {
148  if (HasStationReservation(tile)) return true;
149  }
150  return false;
151  }
152 
154  inline int ReservationCost(Node &n, TileIndex tile, Trackdir trackdir, int skipped)
155  {
156  if (n.m_num_signals_passed >= m_sig_look_ahead_costs.size() / 2) return 0;
157  if (!IsPbsSignal(n.m_last_signal_type)) return 0;
158 
159  if (IsRailStationTile(tile) && IsAnyStationTileReserved(tile, trackdir, skipped)) {
160  return Yapf().PfGetSettings().rail_pbs_station_penalty * (skipped + 1);
161  } else if (TrackOverlapsTracks(GetReservedTrackbits(tile), TrackdirToTrack(trackdir))) {
162  int cost = Yapf().PfGetSettings().rail_pbs_cross_penalty;
163  if (!IsDiagonalTrackdir(trackdir)) cost = (cost * YAPF_TILE_CORNER_LENGTH) / YAPF_TILE_LENGTH;
164  return cost * (skipped + 1);
165  }
166  return 0;
167  }
168 
169  int SignalCost(Node &n, TileIndex tile, Trackdir trackdir)
170  {
171  int cost = 0;
172  /* if there is one-way signal in the opposite direction, then it is not our way */
173  if (IsTileType(tile, MP_RAILWAY)) {
174  bool has_signal_against = HasSignalOnTrackdir(tile, ReverseTrackdir(trackdir));
175  bool has_signal_along = HasSignalOnTrackdir(tile, trackdir);
176  if (has_signal_against && !has_signal_along && IsOnewaySignal(tile, TrackdirToTrack(trackdir))) {
177  /* one-way signal in opposite direction */
178  n.m_segment->m_end_segment_reason |= ESRB_DEAD_END;
179  } else {
180  if (has_signal_along) {
181  SignalState sig_state = GetSignalStateByTrackdir(tile, trackdir);
182  SignalType sig_type = GetSignalType(tile, TrackdirToTrack(trackdir));
183 
184  n.m_last_signal_type = sig_type;
185 
186  /* cache the look-ahead polynomial constant only if we didn't pass more signals than the look-ahead limit is */
187  int look_ahead_cost = (n.m_num_signals_passed < m_sig_look_ahead_costs.size()) ? m_sig_look_ahead_costs[n.m_num_signals_passed] : 0;
188  if (sig_state != SIGNAL_STATE_RED) {
189  /* green signal */
190  n.flags_u.flags_s.m_last_signal_was_red = false;
191  /* negative look-ahead red-signal penalties would cause problems later, so use them as positive penalties for green signal */
192  if (look_ahead_cost < 0) {
193  /* add its negation to the cost */
194  cost -= look_ahead_cost;
195  }
196  } else {
197  /* we have a red signal in our direction
198  * was it first signal which is two-way? */
199  if (!IsPbsSignal(sig_type) && Yapf().TreatFirstRedTwoWaySignalAsEOL() && n.flags_u.flags_s.m_choice_seen && has_signal_against && n.m_num_signals_passed == 0) {
200  /* yes, the first signal is two-way red signal => DEAD END. Prune this branch... */
201  Yapf().PruneIntermediateNodeBranch(&n);
202  n.m_segment->m_end_segment_reason |= ESRB_DEAD_END;
203  Yapf().m_stopped_on_first_two_way_signal = true;
204  return -1;
205  }
206  n.m_last_red_signal_type = sig_type;
207  n.flags_u.flags_s.m_last_signal_was_red = true;
208 
209  /* look-ahead signal penalty */
210  if (!IsPbsSignal(sig_type) && look_ahead_cost > 0) {
211  /* add the look ahead penalty only if it is positive */
212  cost += look_ahead_cost;
213  }
214 
215  /* special signal penalties */
216  if (n.m_num_signals_passed == 0) {
217  switch (sig_type) {
218  case SIGTYPE_COMBO:
219  case SIGTYPE_EXIT: cost += Yapf().PfGetSettings().rail_firstred_exit_penalty; break; // first signal is red pre-signal-exit
220  case SIGTYPE_BLOCK:
221  case SIGTYPE_ENTRY: cost += Yapf().PfGetSettings().rail_firstred_penalty; break;
222  default: break;
223  }
224  }
225  }
226 
227  n.m_num_signals_passed++;
228  n.m_segment->m_last_signal_tile = tile;
229  n.m_segment->m_last_signal_td = trackdir;
230  }
231 
232  if (has_signal_against && IsPbsSignal(GetSignalType(tile, TrackdirToTrack(trackdir)))) {
233  cost += n.m_num_signals_passed < Yapf().PfGetSettings().rail_look_ahead_max_signals ? Yapf().PfGetSettings().rail_pbs_signal_back_penalty : 0;
234  }
235  }
236  }
237  return cost;
238  }
239 
240  inline int PlatformLengthPenalty(int platform_length)
241  {
242  int cost = 0;
243  const Train *v = Yapf().GetVehicle();
244  assert(v != nullptr);
245  assert(v->type == VEH_TRAIN);
246  assert(v->gcache.cached_total_length != 0);
247  int missing_platform_length = CeilDiv(v->gcache.cached_total_length, TILE_SIZE) - platform_length;
248  if (missing_platform_length < 0) {
249  /* apply penalty for longer platform than needed */
250  cost += Yapf().PfGetSettings().rail_longer_platform_penalty + Yapf().PfGetSettings().rail_longer_platform_per_tile_penalty * -missing_platform_length;
251  } else if (missing_platform_length > 0) {
252  /* apply penalty for shorter platform than needed */
253  cost += Yapf().PfGetSettings().rail_shorter_platform_penalty + Yapf().PfGetSettings().rail_shorter_platform_per_tile_penalty * missing_platform_length;
254  }
255  return cost;
256  }
257 
258 public:
259  inline void SetMaxCost(int max_cost)
260  {
261  m_max_cost = max_cost;
262  }
263 
269  inline bool PfCalcCost(Node &n, const TrackFollower *tf)
270  {
271  assert(!n.flags_u.flags_s.m_targed_seen);
272  assert(tf->m_new_tile == n.m_key.m_tile);
273  assert((HasTrackdir(tf->m_new_td_bits, n.m_key.m_td)));
274 
275  /* Does the node have some parent node? */
276  bool has_parent = (n.m_parent != nullptr);
277 
278  /* Do we already have a cached segment? */
279  CachedData &segment = *n.m_segment;
280  bool is_cached_segment = (segment.m_cost >= 0);
281 
282  int parent_cost = has_parent ? n.m_parent->m_cost : 0;
283 
284  /* Each node cost contains 2 or 3 main components:
285  * 1. Transition cost - cost of the move from previous node (tile):
286  * - curve cost (or zero for straight move)
287  * 2. Tile cost:
288  * - base tile cost
289  * - YAPF_TILE_LENGTH for diagonal tiles
290  * - YAPF_TILE_CORNER_LENGTH for non-diagonal tiles
291  * - tile penalties
292  * - tile slope penalty (upward slopes)
293  * - red signal penalty
294  * - level crossing penalty
295  * - speed-limit penalty (bridges)
296  * - station platform penalty
297  * - penalty for reversing in the depot
298  * - etc.
299  * 3. Extra cost (applies to the last node only)
300  * - last red signal penalty
301  * - penalty for too long or too short platform on the destination station
302  */
303  int transition_cost = 0;
304  int extra_cost = 0;
305 
306  /* Segment: one or more tiles connected by contiguous tracks of the same type.
307  * Each segment cost includes 'Tile cost' for all its tiles (including the first
308  * and last), and the 'Transition cost' between its tiles. The first transition
309  * cost of segment entry (move from the 'parent' node) is not included!
310  */
311  int segment_entry_cost = 0;
312  int segment_cost = 0;
313 
314  const Train *v = Yapf().GetVehicle();
315 
316  /* start at n.m_key.m_tile / n.m_key.m_td and walk to the end of segment */
317  TILE cur(n.m_key.m_tile, n.m_key.m_td);
318 
319  /* the previous tile will be needed for transition cost calculations */
320  TILE prev = !has_parent ? TILE() : TILE(n.m_parent->GetLastTile(), n.m_parent->GetLastTrackdir());
321 
322  EndSegmentReasonBits end_segment_reason = ESRB_NONE;
323 
324  TrackFollower tf_local(v, Yapf().GetCompatibleRailTypes());
325 
326  if (!has_parent) {
327  /* We will jump to the middle of the cost calculator assuming that segment cache is not used. */
328  assert(!is_cached_segment);
329  /* Skip the first transition cost calculation. */
330  goto no_entry_cost;
331  }
332 
333  for (;;) {
334  /* Transition cost (cost of the move from previous tile) */
335  transition_cost = Yapf().CurveCost(prev.td, cur.td);
336  transition_cost += Yapf().SwitchCost(prev.tile, cur.tile, TrackdirToExitdir(prev.td));
337 
338  /* First transition cost counts against segment entry cost, other transitions
339  * inside segment will come to segment cost (and will be cached) */
340  if (segment_cost == 0) {
341  /* We just entered the loop. First transition cost goes to segment entry cost)*/
342  segment_entry_cost = transition_cost;
343  transition_cost = 0;
344 
345  /* It is the right time now to look if we can reuse the cached segment cost. */
346  if (is_cached_segment) {
347  /* Yes, we already know the segment cost. */
348  segment_cost = segment.m_cost;
349  /* We know also the reason why the segment ends. */
350  end_segment_reason = segment.m_end_segment_reason;
351  /* We will need also some information about the last signal (if it was red). */
352  if (segment.m_last_signal_tile != INVALID_TILE) {
353  assert(HasSignalOnTrackdir(segment.m_last_signal_tile, segment.m_last_signal_td));
354  SignalState sig_state = GetSignalStateByTrackdir(segment.m_last_signal_tile, segment.m_last_signal_td);
355  bool is_red = (sig_state == SIGNAL_STATE_RED);
356  n.flags_u.flags_s.m_last_signal_was_red = is_red;
357  if (is_red) {
358  n.m_last_red_signal_type = GetSignalType(segment.m_last_signal_tile, TrackdirToTrack(segment.m_last_signal_td));
359  }
360  }
361  /* No further calculation needed. */
362  cur = TILE(n.GetLastTile(), n.GetLastTrackdir());
363  break;
364  }
365  } else {
366  /* Other than first transition cost count as the regular segment cost. */
367  segment_cost += transition_cost;
368  }
369 
370 no_entry_cost: // jump here at the beginning if the node has no parent (it is the first node)
371 
372  /* All other tile costs will be calculated here. */
373  segment_cost += Yapf().OneTileCost(cur.tile, cur.td);
374 
375  /* If we skipped some tunnel/bridge/station tiles, add their base cost */
376  segment_cost += YAPF_TILE_LENGTH * tf->m_tiles_skipped;
377 
378  /* Slope cost. */
379  segment_cost += Yapf().SlopeCost(cur.tile, cur.td);
380 
381  /* Signal cost (routine can modify segment data). */
382  segment_cost += Yapf().SignalCost(n, cur.tile, cur.td);
383 
384  /* Reserved tiles. */
385  segment_cost += Yapf().ReservationCost(n, cur.tile, cur.td, tf->m_tiles_skipped);
386 
387  end_segment_reason = segment.m_end_segment_reason;
388 
389  /* Tests for 'potential target' reasons to close the segment. */
390  if (cur.tile == prev.tile) {
391  /* Penalty for reversing in a depot. */
392  assert(IsRailDepot(cur.tile));
393  segment_cost += Yapf().PfGetSettings().rail_depot_reverse_penalty;
394 
395  } else if (IsRailDepotTile(cur.tile)) {
396  /* We will end in this pass (depot is possible target) */
397  end_segment_reason |= ESRB_DEPOT;
398 
399  } else if (cur.tile_type == MP_STATION && IsRailWaypoint(cur.tile)) {
400  if (v->current_order.IsType(OT_GOTO_WAYPOINT) &&
401  GetStationIndex(cur.tile) == v->current_order.GetDestination() &&
403  /* This waypoint is our destination; maybe this isn't an unreserved
404  * one, so check that and if so see that as the last signal being
405  * red. This way waypoints near stations should work better. */
406  CFollowTrackRail ft(v);
407  TileIndex t = cur.tile;
408  Trackdir td = cur.td;
409  /* Arbitrary maximum tiles to follow to avoid infinite loops. */
410  uint max_tiles = 20;
411  while (ft.Follow(t, td)) {
412  assert(t != ft.m_new_tile);
413  t = ft.m_new_tile;
414  if (t == cur.tile || --max_tiles == 0) {
415  /* We looped back on ourself or found another loop, bail out. */
416  td = INVALID_TRACKDIR;
417  break;
418  }
420  /* We encountered a junction; it's going to be too complex to
421  * handle this perfectly, so just bail out. There is no simple
422  * free path, so try the other possibilities. */
423  td = INVALID_TRACKDIR;
424  break;
425  }
427  /* If this is a safe waiting position we're done searching for it */
428  if (IsSafeWaitingPosition(v, t, td, true, _settings_game.pf.forbid_90_deg)) break;
429  }
430 
431  /* In the case this platform is (possibly) occupied we add penalty so the
432  * other platforms of this waypoint are evaluated as well, i.e. we assume
433  * that there is a red signal in the waypoint when it's occupied. */
434  if (td == INVALID_TRACKDIR ||
437  extra_cost += Yapf().PfGetSettings().rail_lastred_penalty;
438  }
439  }
440  /* Waypoint is also a good reason to finish. */
441  end_segment_reason |= ESRB_WAYPOINT;
442 
443  } else if (tf->m_is_station) {
444  /* Station penalties. */
445  uint platform_length = tf->m_tiles_skipped + 1;
446  /* We don't know yet if the station is our target or not. Act like
447  * if it is pass-through station (not our destination). */
448  segment_cost += Yapf().PfGetSettings().rail_station_penalty * platform_length;
449  /* We will end in this pass (station is possible target) */
450  end_segment_reason |= ESRB_STATION;
451 
452  } else if (TrackFollower::DoTrackMasking() && cur.tile_type == MP_RAILWAY) {
453  /* Searching for a safe tile? */
454  if (HasSignalOnTrackdir(cur.tile, cur.td) && !IsPbsSignal(GetSignalType(cur.tile, TrackdirToTrack(cur.td)))) {
455  end_segment_reason |= ESRB_SAFE_TILE;
456  }
457  }
458 
459  /* Apply min/max speed penalties only when inside the look-ahead radius. Otherwise
460  * it would cause desync in MP. */
461  if (n.m_num_signals_passed < m_sig_look_ahead_costs.size())
462  {
463  int min_speed = 0;
464  int max_speed = tf->GetSpeedLimit(&min_speed);
465  int max_veh_speed = std::min<int>(v->GetDisplayMaxSpeed(), v->current_order.GetMaxSpeed());
466  if (max_speed < max_veh_speed) {
467  extra_cost += YAPF_TILE_LENGTH * (max_veh_speed - max_speed) * (4 + tf->m_tiles_skipped) / max_veh_speed;
468  }
469  if (min_speed > max_veh_speed) {
470  extra_cost += YAPF_TILE_LENGTH * (min_speed - max_veh_speed);
471  }
472  }
473 
474  /* Finish if we already exceeded the maximum path cost (i.e. when
475  * searching for the nearest depot). */
476  if (m_max_cost > 0 && (parent_cost + segment_entry_cost + segment_cost) > m_max_cost) {
477  end_segment_reason |= ESRB_PATH_TOO_LONG;
478  }
479 
480  /* Move to the next tile/trackdir. */
481  tf = &tf_local;
482  tf_local.Init(v, Yapf().GetCompatibleRailTypes());
483 
484  if (!tf_local.Follow(cur.tile, cur.td)) {
485  assert(tf_local.m_err != TrackFollower::EC_NONE);
486  /* Can't move to the next tile (EOL?). */
487  if (tf_local.m_err == TrackFollower::EC_RAIL_ROAD_TYPE) {
488  end_segment_reason |= ESRB_RAIL_TYPE;
489  } else {
490  end_segment_reason |= ESRB_DEAD_END;
491  }
492 
493  if (TrackFollower::DoTrackMasking() && !HasOnewaySignalBlockingTrackdir(cur.tile, cur.td)) {
494  end_segment_reason |= ESRB_SAFE_TILE;
495  }
496  break;
497  }
498 
499  /* Check if the next tile is not a choice. */
500  if (KillFirstBit(tf_local.m_new_td_bits) != TRACKDIR_BIT_NONE) {
501  /* More than one segment will follow. Close this one. */
502  end_segment_reason |= ESRB_CHOICE_FOLLOWS;
503  break;
504  }
505 
506  /* Gather the next tile/trackdir/tile_type/rail_type. */
507  TILE next(tf_local.m_new_tile, (Trackdir)FindFirstBit(tf_local.m_new_td_bits));
508 
509  if (TrackFollower::DoTrackMasking() && IsTileType(next.tile, MP_RAILWAY)) {
510  if (HasSignalOnTrackdir(next.tile, next.td) && IsPbsSignal(GetSignalType(next.tile, TrackdirToTrack(next.td)))) {
511  /* Possible safe tile. */
512  end_segment_reason |= ESRB_SAFE_TILE;
513  } else if (HasSignalOnTrackdir(next.tile, ReverseTrackdir(next.td)) && GetSignalType(next.tile, TrackdirToTrack(next.td)) == SIGTYPE_PBS_ONEWAY) {
514  /* Possible safe tile, but not so good as it's the back of a signal... */
515  end_segment_reason |= ESRB_SAFE_TILE | ESRB_DEAD_END;
516  extra_cost += Yapf().PfGetSettings().rail_lastred_exit_penalty;
517  }
518  }
519 
520  /* Check the next tile for the rail type. */
521  if (next.rail_type != cur.rail_type) {
522  /* Segment must consist from the same rail_type tiles. */
523  end_segment_reason |= ESRB_RAIL_TYPE;
524  break;
525  }
526 
527  /* Avoid infinite looping. */
528  if (next.tile == n.m_key.m_tile && next.td == n.m_key.m_td) {
529  end_segment_reason |= ESRB_INFINITE_LOOP;
530  break;
531  }
532 
533  if (segment_cost > s_max_segment_cost) {
534  /* Potentially in the infinite loop (or only very long segment?). We should
535  * not force it to finish prematurely unless we are on a regular tile. */
536  if (IsTileType(tf->m_new_tile, MP_RAILWAY)) {
537  end_segment_reason |= ESRB_SEGMENT_TOO_LONG;
538  break;
539  }
540  }
541 
542  /* Any other reason bit set? */
543  if (end_segment_reason != ESRB_NONE) {
544  break;
545  }
546 
547  /* For the next loop set new prev and cur tile info. */
548  prev = cur;
549  cur = next;
550 
551  } // for (;;)
552 
553  /* Don't consider path any further it if exceeded max_cost. */
554  if (end_segment_reason & ESRB_PATH_TOO_LONG) return false;
555 
556  bool target_seen = false;
557  if ((end_segment_reason & ESRB_POSSIBLE_TARGET) != ESRB_NONE) {
558  /* Depot, station or waypoint. */
559  if (Yapf().PfDetectDestination(cur.tile, cur.td)) {
560  /* Destination found. */
561  target_seen = true;
562  }
563  }
564 
565  /* Update the segment if needed. */
566  if (!is_cached_segment) {
567  /* Write back the segment information so it can be reused the next time. */
568  segment.m_cost = segment_cost;
569  segment.m_end_segment_reason = end_segment_reason & ESRB_CACHED_MASK;
570  /* Save end of segment back to the node. */
571  n.SetLastTileTrackdir(cur.tile, cur.td);
572  }
573 
574  /* Do we have an excuse why not to continue pathfinding in this direction? */
575  if (!target_seen && (end_segment_reason & ESRB_ABORT_PF_MASK) != ESRB_NONE) {
576  /* Reason to not continue. Stop this PF branch. */
577  return false;
578  }
579 
580  /* Special costs for the case we have reached our target. */
581  if (target_seen) {
582  n.flags_u.flags_s.m_targed_seen = true;
583  /* Last-red and last-red-exit penalties. */
584  if (n.flags_u.flags_s.m_last_signal_was_red) {
585  if (n.m_last_red_signal_type == SIGTYPE_EXIT) {
586  /* last signal was red pre-signal-exit */
587  extra_cost += Yapf().PfGetSettings().rail_lastred_exit_penalty;
588  } else if (!IsPbsSignal(n.m_last_red_signal_type)) {
589  /* Last signal was red, but not exit or path signal. */
590  extra_cost += Yapf().PfGetSettings().rail_lastred_penalty;
591  }
592  }
593 
594  /* Station platform-length penalty. */
595  if ((end_segment_reason & ESRB_STATION) != ESRB_NONE) {
596  const BaseStation *st = BaseStation::GetByTile(n.GetLastTile());
597  assert(st != nullptr);
598  uint platform_length = st->GetPlatformLength(n.GetLastTile(), ReverseDiagDir(TrackdirToExitdir(n.GetLastTrackdir())));
599  /* Reduce the extra cost caused by passing-station penalty (each station receives it in the segment cost). */
600  extra_cost -= Yapf().PfGetSettings().rail_station_penalty * platform_length;
601  /* Add penalty for the inappropriate platform length. */
602  extra_cost += PlatformLengthPenalty(platform_length);
603  }
604  }
605 
606  /* total node cost */
607  n.m_cost = parent_cost + segment_entry_cost + segment_cost + extra_cost;
608 
609  return true;
610  }
611 
612  inline bool CanUseGlobalCache(Node &n) const
613  {
614  return !m_disable_cache
615  && (n.m_parent != nullptr)
616  && (n.m_parent->m_num_signals_passed >= m_sig_look_ahead_costs.size());
617  }
618 
619  inline void ConnectNodeToCachedData(Node &n, CachedData &ci)
620  {
621  n.m_segment = &ci;
622  if (n.m_segment->m_cost < 0) {
623  n.m_segment->m_last_tile = n.m_key.m_tile;
624  n.m_segment->m_last_td = n.m_key.m_td;
625  }
626  }
627 
628  void DisableCache(bool disable)
629  {
630  m_disable_cache = disable;
631  }
632 };
633 
634 #endif /* YAPF_COSTRAIL_HPP */
INVALID_RAILTYPE
@ INVALID_RAILTYPE
Flag for invalid railtype.
Definition: rail_type.h:34
DiagdirReachesTracks
TrackBits DiagdirReachesTracks(DiagDirection diagdir)
Returns all tracks that can be reached when entering a tile from a given (diagonal) direction.
Definition: track_func.h:573
CYapfCostRailT::IsAnyStationTileReserved
bool IsAnyStationTileReserved(TileIndex tile, Trackdir trackdir, int skipped)
Check for a reserved station platform.
Definition: yapf_costrail.hpp:144
Order::IsType
bool IsType(OrderType type) const
Check whether this order is of the given type.
Definition: order_base.h:70
SIGTYPE_EXIT
@ SIGTYPE_EXIT
presignal block exit
Definition: signal_type.h:26
SIGTYPE_BLOCK
@ SIGTYPE_BLOCK
block signal
Definition: signal_type.h:24
PathfinderSettings::forbid_90_deg
bool forbid_90_deg
forbid trains to make 90 deg turns
Definition: settings_type.h:466
SIGTYPE_COMBO
@ SIGTYPE_COMBO
presignal inter-block
Definition: signal_type.h:27
IsOnewaySignal
bool IsOnewaySignal(Tile t, Track track)
One-way signals can't be passed the 'wrong' way.
Definition: rail_map.h:319
CYapfCostRailT::m_max_cost
int m_max_cost
Definition: yapf_costrail.hpp:56
GetReservedTrackbits
TrackBits GetReservedTrackbits(TileIndex t)
Get the reserved trackbits for any tile, regardless of type.
Definition: pbs.cpp:24
CYapfCostRailT::PfCalcCost
bool PfCalcCost(Node &n, const TrackFollower *tf)
Called by YAPF to calculate the cost from the origin to the given node.
Definition: yapf_costrail.hpp:269
BaseStation::GetByTile
static BaseStation * GetByTile(TileIndex tile)
Get the base station belonging to a specific tile.
Definition: base_station_base.h:166
Order::GetDestination
DestinationID GetDestination() const
Gets the destination of this order.
Definition: order_base.h:103
TrackdirToTrack
Track TrackdirToTrack(Trackdir trackdir)
Returns the Track that a given Trackdir represents.
Definition: track_func.h:262
TrackdirToExitdir
DiagDirection TrackdirToExitdir(Trackdir trackdir)
Maps a trackdir to the (4-way) direction the tile is exited when following that trackdir.
Definition: track_func.h:439
RailType
RailType
Enumeration for all possible railtypes.
Definition: rail_type.h:27
VEH_TRAIN
@ VEH_TRAIN
Train vehicle type.
Definition: vehicle_type.h:24
TrackOverlapsTracks
bool TrackOverlapsTracks(TrackBits tracks, Track track)
Check if a given track is contained within or overlaps some other tracks.
Definition: track_func.h:662
INVALID_TILE
constexpr TileIndex INVALID_TILE
The very nice invalid tile marker.
Definition: tile_type.h:95
MP_RAILWAY
@ MP_RAILWAY
A railway.
Definition: tile_type.h:49
IsRailStationTile
bool IsRailStationTile(Tile t)
Is this tile a station tile and a rail station?
Definition: station_map.h:102
TILE_SIZE
static const uint TILE_SIZE
Tile size in world coordinates.
Definition: tile_type.h:15
DiagDirection
DiagDirection
Enumeration for diagonal directions.
Definition: direction_type.h:73
SignalState
SignalState
These are states in which a signal can be.
Definition: signal_type.h:42
CeilDiv
constexpr uint CeilDiv(uint a, uint b)
Computes ceil(a / b) for non-negative a and b.
Definition: math_func.hpp:320
CYapfCostRailT
Definition: yapf_costrail.hpp:17
SpecializedStation< Waypoint, true >::Get
static Waypoint * Get(size_t index)
Gets station with given index.
Definition: base_station_base.h:254
IsLevelCrossing
bool IsLevelCrossing(Tile t)
Return whether a tile is a level crossing.
Definition: road_map.h:85
StrongType::Typedef< uint32_t, struct TileIndexTag, StrongType::Compare, StrongType::Integer, StrongType::Compatible< int32_t >, StrongType::Compatible< int64_t > >
NextTrackdir
Trackdir NextTrackdir(Trackdir trackdir)
Maps a trackdir to the trackdir that you will end up on if you go straight ahead.
Definition: track_func.h:403
CYapfCostRailT::ReservationCost
int ReservationCost(Node &n, TileIndex tile, Trackdir trackdir, int skipped)
The cost for reserved tiles, including skipped ones.
Definition: yapf_costrail.hpp:154
MP_ROAD
@ MP_ROAD
A tile with road (or tram tracks)
Definition: tile_type.h:50
KillFirstBit
constexpr T KillFirstBit(T value)
Clear the first bit in an integer.
Definition: bitmath_func.hpp:250
SIGTYPE_PBS_ONEWAY
@ SIGTYPE_PBS_ONEWAY
no-entry signal
Definition: signal_type.h:29
GetTileType
static debug_inline TileType GetTileType(Tile tile)
Get the tiletype of a given tile.
Definition: tile_map.h:96
TRACK_BIT_NONE
@ TRACK_BIT_NONE
No track.
Definition: track_type.h:36
CYapfCostRailT::Key
Node::Key Key
key to hash tables
Definition: yapf_costrail.hpp:22
CYapfCostBase
Base implementation for cost accounting.
Definition: yapf_costbase.hpp:14
CYapfCostBase::stSlopeCost
static bool stSlopeCost(TileIndex tile, Trackdir td)
Does the given track direction on the given tile yield an uphill penalty?
Definition: yapf_costbase.hpp:21
GameSettings::pf
PathfinderSettings pf
settings for all pathfinders
Definition: settings_type.h:600
IsRailDepot
static debug_inline bool IsRailDepot(Tile t)
Is this rail tile a rail depot?
Definition: rail_map.h:95
IsWaitingPositionFree
bool IsWaitingPositionFree(const Train *v, TileIndex tile, Trackdir trackdir, bool forbid_90deg)
Check if a safe position is free.
Definition: pbs.cpp:426
Waypoint::IsSingleTile
bool IsSingleTile() const
Is this a single tile waypoint?
Definition: waypoint_base.h:62
YAPF_TILE_LENGTH
static const int YAPF_TILE_LENGTH
Length (penalty) of one tile with YAPF.
Definition: pathfinder_type.h:16
SIGNAL_STATE_RED
@ SIGNAL_STATE_RED
The signal is red.
Definition: signal_type.h:43
TRACKDIR_BIT_NONE
@ TRACKDIR_BIT_NONE
No track build.
Definition: track_type.h:99
ReverseDiagDir
DiagDirection ReverseDiagDir(DiagDirection d)
Returns the reverse direction of the given DiagDirection.
Definition: direction_func.h:118
Vehicle::current_order
Order current_order
The current order (+ status, like: loading)
Definition: vehicle_base.h:356
GroundVehicle::gcache
GroundVehicleCache gcache
Cache of often calculated values.
Definition: ground_vehicle.hpp:83
ReverseTrackdir
Trackdir ReverseTrackdir(Trackdir trackdir)
Maps a trackdir to the reverse trackdir.
Definition: track_func.h:247
_settings_game
GameSettings _settings_game
Game settings of a running game or the scenario editor.
Definition: settings.cpp:57
CYapfCostRailT::Yapf
Tpf & Yapf()
to access inherited path finder
Definition: yapf_costrail.hpp:80
Train
'Train' is either a loco or a wagon.
Definition: train.h:89
RemoveFirstTrackdir
Trackdir RemoveFirstTrackdir(TrackdirBits *trackdirs)
Removes first Trackdir from TrackdirBits and returns it.
Definition: track_func.h:156
IsSafeWaitingPosition
bool IsSafeWaitingPosition(const Train *v, TileIndex tile, Trackdir trackdir, bool include_line_end, bool forbid_90deg)
Determine whether a certain track on a tile is a safe position to end a path.
Definition: pbs.cpp:380
GetSignalStateByTrackdir
SignalState GetSignalStateByTrackdir(Tile tile, Trackdir trackdir)
Gets the state of the signal along the given trackdir.
Definition: rail_map.h:438
TileIndexDiff
int32_t TileIndexDiff
An offset value between two tiles.
Definition: map_func.h:376
HasOnewaySignalBlockingTrackdir
bool HasOnewaySignalBlockingTrackdir(Tile tile, Trackdir td)
Is a one-way signal blocking the trackdir? A one-way signal on the trackdir against will block,...
Definition: rail_map.h:475
CYapfCostRailT::Tpf
Types::Tpf Tpf
the pathfinder class (derived from THIS class)
Definition: yapf_costrail.hpp:19
SIGTYPE_ENTRY
@ SIGTYPE_ENTRY
presignal block entry
Definition: signal_type.h:25
CYapfCostRailT::Node
Types::NodeList::Titem Node
this will be our node type
Definition: yapf_costrail.hpp:21
IsPlainRailTile
static debug_inline bool IsPlainRailTile(Tile t)
Checks whether the tile is a rail tile or rail tile with signals.
Definition: rail_map.h:60
GetTrackBits
TrackBits GetTrackBits(Tile tile)
Gets the track bits of the given tile.
Definition: rail_map.h:136
TileOffsByDiagDir
TileIndexDiff TileOffsByDiagDir(DiagDirection dir)
Convert a DiagDirection to a TileIndexDiff.
Definition: map_func.h:565
MP_VOID
@ MP_VOID
Invisible tiles at the SW and SE border.
Definition: tile_type.h:55
Trackdir
Trackdir
Enumeration for tracks and directions.
Definition: track_type.h:67
IsRailDepotTile
static debug_inline bool IsRailDepotTile(Tile t)
Is this tile rail tile and a rail depot?
Definition: rail_map.h:105
TileType
TileType
The different types of tiles.
Definition: tile_type.h:47
GetStationIndex
StationID GetStationIndex(Tile t)
Get StationID from a tile.
Definition: station_map.h:28
HasStationReservation
bool HasStationReservation(Tile t)
Get the reservation state of the rail station.
Definition: station_map.h:552
GetTileRailType
RailType GetTileRailType(Tile tile)
Return the rail type of tile, or INVALID_RAILTYPE if this is no rail tile.
Definition: rail.cpp:155
MP_STATION
@ MP_STATION
A tile of a station.
Definition: tile_type.h:53
BaseStation
Base class for all station-ish types.
Definition: base_station_base.h:59
IsDiagonalTrackdir
bool IsDiagonalTrackdir(Trackdir trackdir)
Checks if a given Trackdir is diagonal.
Definition: track_func.h:631
Order::GetMaxSpeed
uint16_t GetMaxSpeed() const
Get the maxmimum speed in km-ish/h a vehicle is allowed to reach on the way to the destination.
Definition: order_base.h:201
CFollowTrackT::m_new_tile
TileIndex m_new_tile
the new tile (the vehicle has entered)
Definition: follow_track.hpp:43
CFollowTrackT::Follow
bool Follow(TileIndex old_tile, Trackdir old_td)
main follower routine.
Definition: follow_track.hpp:119
CFollowTrackT::m_new_td_bits
TrackdirBits m_new_td_bits
the new set of available trackdirs
Definition: follow_track.hpp:44
YAPF_TILE_CORNER_LENGTH
static const int YAPF_TILE_CORNER_LENGTH
Length (penalty) of a corner with YAPF.
Definition: pathfinder_type.h:19
SignalType
SignalType
Type of signal, i.e.
Definition: signal_type.h:23
IsTileType
static debug_inline bool IsTileType(Tile tile, TileType type)
Checks if a tile is a given tiletype.
Definition: tile_map.h:150
IsRailWaypoint
bool IsRailWaypoint(Tile t)
Is this station tile a rail waypoint?
Definition: station_map.h:113
TrackdirCrossesTrackdirs
TrackdirBits TrackdirCrossesTrackdirs(Trackdir trackdir)
Maps a trackdir to all trackdirs that make 90 deg turns with it.
Definition: track_func.h:606
BaseVehicle::type
VehicleType type
Type of vehicle.
Definition: vehicle_type.h:51
LinkGraph::BaseNode
Node of the link graph.
Definition: linkgraph.h:90
BaseStation::GetPlatformLength
virtual uint GetPlatformLength(TileIndex tile) const =0
Obtain the length of a platform.
HasTrackdir
bool HasTrackdir(TrackdirBits trackdirs, Trackdir trackdir)
Checks whether a TrackdirBits has a given Trackdir.
Definition: track_func.h:340
CYapfCostRailT::TILE
Definition: yapf_costrail.hpp:28
IsValidTrackdir
bool IsValidTrackdir(Trackdir trackdir)
Checks if a Trackdir is valid for non-road vehicles.
Definition: track_func.h:52
Train::GetDisplayMaxSpeed
int GetDisplayMaxSpeed() const override
Gets the maximum speed in km-ish/h that can be sent into SetDParam for string processing.
Definition: train.h:119
INVALID_TRACKDIR
@ INVALID_TRACKDIR
Flag for an invalid trackdir.
Definition: track_type.h:86
GroundVehicleCache::cached_total_length
uint16_t cached_total_length
Length of the whole vehicle (valid only for the first engine).
Definition: ground_vehicle.hpp:43
CYapfCostRailT::OneTileCost
int OneTileCost(TileIndex &tile, Trackdir trackdir)
Return one tile cost (base cost + level crossing penalty).
Definition: yapf_costrail.hpp:119
CFollowTrackT
Track follower helper template class (can serve pathfinders and vehicle controllers).
Definition: follow_track.hpp:28
HasSignalOnTrackdir
bool HasSignalOnTrackdir(Tile tile, Trackdir trackdir)
Checks for the presence of signals along the given trackdir on the given rail tile.
Definition: rail_map.h:426
FindFirstBit
constexpr uint8_t FindFirstBit(T x)
Search the first set bit in a value.
Definition: bitmath_func.hpp:213