OpenTTD Source  20240917-master-g9ab0a47812
yapf_rail.cpp
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 #include "../../stdafx.h"
11 
12 #include "yapf.hpp"
13 #include "yapf_cache.h"
14 #include "yapf_node_rail.hpp"
15 #include "yapf_costrail.hpp"
16 #include "yapf_destrail.hpp"
17 #include "../../viewport_func.h"
18 #include "../../newgrf_station.h"
19 
20 #include "../../safeguards.h"
21 
22 template <typename Tpf> void DumpState(Tpf &pf1, Tpf &pf2)
23 {
24  DumpTarget dmp1, dmp2;
25  pf1.DumpBase(dmp1);
26  pf2.DumpBase(dmp2);
27  auto f1 = FileHandle::Open("yapf1.txt", "wt");
28  auto f2 = FileHandle::Open("yapf2.txt", "wt");
29  assert(f1.has_value());
30  assert(f2.has_value());
31  fwrite(dmp1.m_out.c_str(), 1, dmp1.m_out.size(), *f1);
32  fwrite(dmp2.m_out.c_str(), 1, dmp2.m_out.size(), *f2);
33 }
34 
35 template <class Types>
37 {
38 public:
39  typedef typename Types::Tpf Tpf;
40  typedef typename Types::TrackFollower TrackFollower;
41  typedef typename Types::NodeList::Titem Node;
42 
43 protected:
45  inline Tpf &Yapf()
46  {
47  return *static_cast<Tpf *>(this);
48  }
49 
50 private:
57 
58  std::vector<std::pair<TileIndex, Trackdir>> m_signals_set_to_red;
59 
60  bool FindSafePositionProc(TileIndex tile, Trackdir td)
61  {
62  if (IsSafeWaitingPosition(Yapf().GetVehicle(), tile, td, true, !TrackFollower::Allow90degTurns())) {
63  m_res_dest = tile;
64  m_res_dest_td = td;
65  return false; // Stop iterating segment
66  }
67  return true;
68  }
69 
72  {
73  TileIndex start = tile;
74  TileIndexDiff diff = TileOffsByDiagDir(dir);
75 
76  do {
77  if (HasStationReservation(tile)) return false;
78  SetRailStationReservation(tile, true);
79  MarkTileDirtyByTile(tile);
80  tile = TileAdd(tile, diff);
81  } while (IsCompatibleTrainStationTile(tile, start) && tile != m_origin_tile);
82 
84 
85  return true;
86  }
87 
90  {
91  Trackdir rev_td = ReverseTrackdir(td);
92  if (IsRailStationTile(tile)) {
93  if (!ReserveRailStationPlatform(tile, TrackdirToExitdir(rev_td))) {
94  /* Platform could not be reserved, undo. */
95  m_res_fail_tile = tile;
96  m_res_fail_td = td;
97  }
98  } else {
99  if (!TryReserveRailTrack(tile, TrackdirToTrack(td))) {
100  /* Tile couldn't be reserved, undo. */
101  m_res_fail_tile = tile;
102  m_res_fail_td = td;
103  return false;
104  }
105 
106  /* Green path signal opposing the path? Turn to red. */
107  if (HasPbsSignalOnTrackdir(tile, rev_td) && GetSignalStateByTrackdir(tile, rev_td) == SIGNAL_STATE_GREEN) {
108  m_signals_set_to_red.emplace_back(tile, rev_td);
110  MarkTileDirtyByTile(tile);
111  }
112  }
113 
114  return tile != m_res_dest || td != m_res_dest_td;
115  }
116 
119  {
120  if (IsRailStationTile(tile)) {
121  TileIndex start = tile;
123  while ((tile != m_res_fail_tile || td != m_res_fail_td) && IsCompatibleTrainStationTile(tile, start)) {
124  SetRailStationReservation(tile, false);
125  tile = TileAdd(tile, diff);
126  }
127  } else if (tile != m_res_fail_tile || td != m_res_fail_td) {
129  }
130  return (tile != m_res_dest || td != m_res_dest_td) && (tile != m_res_fail_tile || td != m_res_fail_td);
131  }
132 
133 public:
135  inline void SetReservationTarget(Node *node, TileIndex tile, Trackdir td)
136  {
137  m_res_node = node;
138  m_res_dest = tile;
139  m_res_dest_td = td;
140  }
141 
143  inline void FindSafePositionOnNode(Node *node)
144  {
145  assert(node->m_parent != nullptr);
146 
147  /* We will never pass more than two signals, no need to check for a safe tile. */
148  if (node->m_parent->m_num_signals_passed >= 2) return;
149 
150  if (!node->IterateTiles(Yapf().GetVehicle(), Yapf(), *this, &CYapfReserveTrack<Types>::FindSafePositionProc)) {
151  m_res_node = node;
152  }
153  }
154 
156  bool TryReservePath(PBSTileInfo *target, TileIndex origin)
157  {
159  m_origin_tile = origin;
160 
161  if (target != nullptr) {
162  target->tile = m_res_dest;
163  target->trackdir = m_res_dest_td;
164  target->okay = false;
165  }
166 
167  /* Don't bother if the target is reserved. */
168  if (!IsWaitingPositionFree(Yapf().GetVehicle(), m_res_dest, m_res_dest_td)) return false;
169 
170  m_signals_set_to_red.clear();
171  for (Node *node = m_res_node; node->m_parent != nullptr; node = node->m_parent) {
172  node->IterateTiles(Yapf().GetVehicle(), Yapf(), *this, &CYapfReserveTrack<Types>::ReserveSingleTrack);
173  if (m_res_fail_tile != INVALID_TILE) {
174  /* Reservation failed, undo. */
175  Node *fail_node = m_res_node;
176  TileIndex stop_tile = m_res_fail_tile;
177  do {
178  /* If this is the node that failed, stop at the failed tile. */
179  m_res_fail_tile = fail_node == node ? stop_tile : INVALID_TILE;
180  fail_node->IterateTiles(Yapf().GetVehicle(), Yapf(), *this, &CYapfReserveTrack<Types>::UnreserveSingleTrack);
181  } while (fail_node != node && (fail_node = fail_node->m_parent) != nullptr);
182 
183  /* Re-instate green path signals we turned to red. */
184  for (auto [sig_tile, td] : m_signals_set_to_red) {
186  }
187 
188  return false;
189  }
190  }
191 
192  if (target != nullptr) target->okay = true;
193 
194  if (Yapf().CanUseGlobalCache(*m_res_node)) {
196  }
197 
198  return true;
199  }
200 };
201 
202 template <class Types>
204 {
205 public:
206  typedef typename Types::Tpf Tpf;
207  typedef typename Types::TrackFollower TrackFollower;
208  typedef typename Types::NodeList::Titem Node;
209  typedef typename Node::Key Key;
210 
211 protected:
213  inline Tpf &Yapf()
214  {
215  return *static_cast<Tpf *>(this);
216  }
217 
218 public:
224  inline void PfFollowNode(Node &old_node)
225  {
226  TrackFollower F(Yapf().GetVehicle());
227  if (F.Follow(old_node.GetLastTile(), old_node.GetLastTrackdir())) {
228  Yapf().AddMultipleNodes(&old_node, F);
229  }
230  }
231 
233  inline char TransportTypeChar() const
234  {
235  return 't';
236  }
237 
238  static FindDepotData stFindNearestDepotTwoWay(const Train *v, TileIndex t1, Trackdir td1, TileIndex t2, Trackdir td2, int max_penalty, int reverse_penalty)
239  {
240  Tpf pf1;
241  /*
242  * With caching enabled it simply cannot get a reliable result when you
243  * have limited the distance a train may travel. This means that the
244  * cached result does not match uncached result in all cases and that
245  * causes desyncs. So disable caching when finding for a depot that is
246  * nearby. This only happens with automatic servicing of vehicles,
247  * so it will only impact performance when you do not manually set
248  * depot orders and you do not disable automatic servicing.
249  */
250  if (max_penalty != 0) pf1.DisableCache(true);
251  FindDepotData result1 = pf1.FindNearestDepotTwoWay(v, t1, td1, t2, td2, max_penalty, reverse_penalty);
252 
253  if (_debug_desync_level >= 2) {
254  Tpf pf2;
255  pf2.DisableCache(true);
256  FindDepotData result2 = pf2.FindNearestDepotTwoWay(v, t1, td1, t2, td2, max_penalty, reverse_penalty);
257  if (result1.tile != result2.tile || (result1.reverse != result2.reverse)) {
258  Debug(desync, 2, "warning: FindNearestDepotTwoWay cache mismatch: {} vs {}",
259  result1.tile != INVALID_TILE ? "T" : "F",
260  result2.tile != INVALID_TILE ? "T" : "F");
261  DumpState(pf1, pf2);
262  }
263  }
264 
265  return result1;
266  }
267 
268  inline FindDepotData FindNearestDepotTwoWay(const Train *v, TileIndex t1, Trackdir td1, TileIndex t2, Trackdir td2, int max_penalty, int reverse_penalty)
269  {
270  /* set origin and destination nodes */
271  Yapf().SetOrigin(t1, td1, t2, td2, reverse_penalty, true);
272  Yapf().SetDestination(v);
273  Yapf().SetMaxCost(max_penalty);
274 
275  /* find the best path */
276  if (!Yapf().FindPath(v)) return FindDepotData();
277 
278  /* Some path found. */
279  Node *n = Yapf().GetBestNode();
280 
281  /* walk through the path back to the origin */
282  Node *pNode = n;
283  while (pNode->m_parent != nullptr) {
284  pNode = pNode->m_parent;
285  }
286 
287  /* if the origin node is our front vehicle tile/Trackdir then we didn't reverse
288  * but we can also look at the cost (== 0 -> not reversed, == reverse_penalty -> reversed) */
289  return FindDepotData(n->GetLastTile(), n->m_cost, pNode->m_cost != 0);
290  }
291 };
292 
293 template <class Types>
295 {
296 public:
297  typedef typename Types::Tpf Tpf;
298  typedef typename Types::TrackFollower TrackFollower;
299  typedef typename Types::NodeList::Titem Node;
300  typedef typename Node::Key Key;
301 
302 protected:
304  inline Tpf &Yapf()
305  {
306  return *static_cast<Tpf *>(this);
307  }
308 
309 public:
315  inline void PfFollowNode(Node &old_node)
316  {
317  TrackFollower F(Yapf().GetVehicle(), Yapf().GetCompatibleRailTypes());
318  if (F.Follow(old_node.GetLastTile(), old_node.GetLastTrackdir()) && F.MaskReservedTracks()) {
319  Yapf().AddMultipleNodes(&old_node, F);
320  }
321  }
322 
324  inline char TransportTypeChar() const
325  {
326  return 't';
327  }
328 
329  static bool stFindNearestSafeTile(const Train *v, TileIndex t1, Trackdir td, bool override_railtype)
330  {
331  /* Create pathfinder instance */
332  Tpf pf1;
333  bool result1;
334  if (_debug_desync_level < 2) {
335  result1 = pf1.FindNearestSafeTile(v, t1, td, override_railtype, false);
336  } else {
337  bool result2 = pf1.FindNearestSafeTile(v, t1, td, override_railtype, true);
338  Tpf pf2;
339  pf2.DisableCache(true);
340  result1 = pf2.FindNearestSafeTile(v, t1, td, override_railtype, false);
341  if (result1 != result2) {
342  Debug(desync, 2, "warning: FindSafeTile cache mismatch: {} vs {}", result2 ? "T" : "F", result1 ? "T" : "F");
343  DumpState(pf1, pf2);
344  }
345  }
346 
347  return result1;
348  }
349 
350  bool FindNearestSafeTile(const Train *v, TileIndex t1, Trackdir td, bool override_railtype, bool dont_reserve)
351  {
352  /* Set origin and destination. */
353  Yapf().SetOrigin(t1, td);
354  Yapf().SetDestination(v, override_railtype);
355 
356  bool bFound = Yapf().FindPath(v);
357  if (!bFound) return false;
358 
359  /* Found a destination, set as reservation target. */
360  Node *pNode = Yapf().GetBestNode();
361  this->SetReservationTarget(pNode, pNode->GetLastTile(), pNode->GetLastTrackdir());
362 
363  /* Walk through the path back to the origin. */
364  Node *pPrev = nullptr;
365  while (pNode->m_parent != nullptr) {
366  pPrev = pNode;
367  pNode = pNode->m_parent;
368 
369  this->FindSafePositionOnNode(pPrev);
370  }
371 
372  return dont_reserve || this->TryReservePath(nullptr, pNode->GetLastTile());
373  }
374 };
375 
376 template <class Types>
377 class CYapfFollowRailT : public CYapfReserveTrack<Types>
378 {
379 public:
380  typedef typename Types::Tpf Tpf;
381  typedef typename Types::TrackFollower TrackFollower;
382  typedef typename Types::NodeList::Titem Node;
383  typedef typename Node::Key Key;
384 
385 protected:
387  inline Tpf &Yapf()
388  {
389  return *static_cast<Tpf *>(this);
390  }
391 
392 public:
398  inline void PfFollowNode(Node &old_node)
399  {
400  TrackFollower F(Yapf().GetVehicle());
401  if (F.Follow(old_node.GetLastTile(), old_node.GetLastTrackdir())) {
402  Yapf().AddMultipleNodes(&old_node, F);
403  }
404  }
405 
407  inline char TransportTypeChar() const
408  {
409  return 't';
410  }
411 
412  static Trackdir stChooseRailTrack(const Train *v, TileIndex tile, DiagDirection enterdir, TrackBits tracks, bool &path_found, bool reserve_track, PBSTileInfo *target, TileIndex *dest)
413  {
414  /* create pathfinder instance */
415  Tpf pf1;
416  Trackdir result1;
417 
418  if (_debug_desync_level < 2) {
419  result1 = pf1.ChooseRailTrack(v, tile, enterdir, tracks, path_found, reserve_track, target, dest);
420  } else {
421  result1 = pf1.ChooseRailTrack(v, tile, enterdir, tracks, path_found, false, nullptr, nullptr);
422  Tpf pf2;
423  pf2.DisableCache(true);
424  Trackdir result2 = pf2.ChooseRailTrack(v, tile, enterdir, tracks, path_found, reserve_track, target, dest);
425  if (result1 != result2) {
426  Debug(desync, 2, "warning: ChooseRailTrack cache mismatch: {} vs {}", result1, result2);
427  DumpState(pf1, pf2);
428  }
429  }
430 
431  return result1;
432  }
433 
434  inline Trackdir ChooseRailTrack(const Train *v, TileIndex, DiagDirection, TrackBits, bool &path_found, bool reserve_track, PBSTileInfo *target, TileIndex *dest)
435  {
436  if (target != nullptr) target->tile = INVALID_TILE;
437  if (dest != nullptr) *dest = INVALID_TILE;
438 
439  /* set origin and destination nodes */
441  Yapf().SetOrigin(origin.tile, origin.trackdir, INVALID_TILE, INVALID_TRACKDIR, 1, true);
442  Yapf().SetDestination(v);
443 
444  /* find the best path */
445  path_found = Yapf().FindPath(v);
446 
447  /* if path not found - return INVALID_TRACKDIR */
448  Trackdir next_trackdir = INVALID_TRACKDIR;
449  Node *pNode = Yapf().GetBestNode();
450  if (pNode != nullptr) {
451  /* reserve till end of path */
452  this->SetReservationTarget(pNode, pNode->GetLastTile(), pNode->GetLastTrackdir());
453 
454  /* path was found or at least suggested
455  * walk through the path back to the origin */
456  Node *pPrev = nullptr;
457  while (pNode->m_parent != nullptr) {
458  pPrev = pNode;
459  pNode = pNode->m_parent;
460 
461  this->FindSafePositionOnNode(pPrev);
462  }
463 
464  /* If the best PF node has no parent, then there is no (valid) best next trackdir to return.
465  * This occurs when the PF is called while the train is already at its destination. */
466  if (pPrev == nullptr) return INVALID_TRACKDIR;
467 
468  /* return trackdir from the best origin node (one of start nodes) */
469  Node &best_next_node = *pPrev;
470  next_trackdir = best_next_node.GetTrackdir();
471 
472  if (reserve_track && path_found) {
473  if (dest != nullptr) *dest = Yapf().GetBestNode()->GetLastTile();
474  this->TryReservePath(target, pNode->GetLastTile());
475  }
476  }
477 
478  /* Treat the path as found if stopped on the first two way signal(s). */
479  path_found |= Yapf().m_stopped_on_first_two_way_signal;
480  return next_trackdir;
481  }
482 
483  static bool stCheckReverseTrain(const Train *v, TileIndex t1, Trackdir td1, TileIndex t2, Trackdir td2, int reverse_penalty)
484  {
485  Tpf pf1;
486  bool result1 = pf1.CheckReverseTrain(v, t1, td1, t2, td2, reverse_penalty);
487 
488  if (_debug_desync_level >= 2) {
489  Tpf pf2;
490  pf2.DisableCache(true);
491  bool result2 = pf2.CheckReverseTrain(v, t1, td1, t2, td2, reverse_penalty);
492  if (result1 != result2) {
493  Debug(desync, 2, "warning: CheckReverseTrain cache mismatch: {} vs {}", result1 ? "T" : "F", result2 ? "T" : "F");
494  DumpState(pf1, pf2);
495  }
496  }
497 
498  return result1;
499  }
500 
501  inline bool CheckReverseTrain(const Train *v, TileIndex t1, Trackdir td1, TileIndex t2, Trackdir td2, int reverse_penalty)
502  {
503  /* create pathfinder instance
504  * set origin and destination nodes */
505  Yapf().SetOrigin(t1, td1, t2, td2, reverse_penalty, false);
506  Yapf().SetDestination(v);
507 
508  /* find the best path */
509  bool bFound = Yapf().FindPath(v);
510 
511  if (!bFound) return false;
512 
513  /* path was found
514  * walk through the path back to the origin */
515  Node *pNode = Yapf().GetBestNode();
516  while (pNode->m_parent != nullptr) {
517  pNode = pNode->m_parent;
518  }
519 
520  /* check if it was reversed origin */
521  Node &best_org_node = *pNode;
522  bool reversed = (best_org_node.m_cost != 0);
523  return reversed;
524  }
525 };
526 
527 template <class Tpf_, class Ttrack_follower, class Tnode_list, template <class Types> class TdestinationT, template <class Types> class TfollowT>
529 {
531 
532  typedef Tpf_ Tpf;
533  typedef Ttrack_follower TrackFollower;
534  typedef Tnode_list NodeList;
535  typedef Train VehicleType;
536  typedef CYapfBaseT<Types> PfBase;
537  typedef TfollowT<Types> PfFollow;
539  typedef TdestinationT<Types> PfDestination;
542 };
543 
544 struct CYapfRail1 : CYapfT<CYapfRail_TypesT<CYapfRail1 , CFollowTrackRail , CRailNodeListTrackDir, CYapfDestinationTileOrStationRailT, CYapfFollowRailT> > {};
545 struct CYapfRail2 : CYapfT<CYapfRail_TypesT<CYapfRail2 , CFollowTrackRailNo90, CRailNodeListTrackDir, CYapfDestinationTileOrStationRailT, CYapfFollowRailT> > {};
546 
547 struct CYapfAnyDepotRail1 : CYapfT<CYapfRail_TypesT<CYapfAnyDepotRail1, CFollowTrackRail , CRailNodeListTrackDir, CYapfDestinationAnyDepotRailT , CYapfFollowAnyDepotRailT> > {};
548 struct CYapfAnyDepotRail2 : CYapfT<CYapfRail_TypesT<CYapfAnyDepotRail2, CFollowTrackRailNo90, CRailNodeListTrackDir, CYapfDestinationAnyDepotRailT , CYapfFollowAnyDepotRailT> > {};
549 
550 struct CYapfAnySafeTileRail1 : CYapfT<CYapfRail_TypesT<CYapfAnySafeTileRail1, CFollowTrackFreeRail , CRailNodeListTrackDir, CYapfDestinationAnySafeTileRailT , CYapfFollowAnySafeTileRailT> > {};
551 struct CYapfAnySafeTileRail2 : CYapfT<CYapfRail_TypesT<CYapfAnySafeTileRail2, CFollowTrackFreeRailNo90, CRailNodeListTrackDir, CYapfDestinationAnySafeTileRailT , CYapfFollowAnySafeTileRailT> > {};
552 
553 
554 Track YapfTrainChooseTrack(const Train *v, TileIndex tile, DiagDirection enterdir, TrackBits tracks, bool &path_found, bool reserve_track, PBSTileInfo *target, TileIndex *dest)
555 {
556  /* default is YAPF type 2 */
557  typedef Trackdir (*PfnChooseRailTrack)(const Train*, TileIndex, DiagDirection, TrackBits, bool&, bool, PBSTileInfo*, TileIndex*);
558  PfnChooseRailTrack pfnChooseRailTrack = &CYapfRail1::stChooseRailTrack;
559 
560  /* check if non-default YAPF type needed */
562  pfnChooseRailTrack = &CYapfRail2::stChooseRailTrack; // Trackdir, forbid 90-deg
563  }
564 
565  Trackdir td_ret = pfnChooseRailTrack(v, tile, enterdir, tracks, path_found, reserve_track, target, dest);
566  return (td_ret != INVALID_TRACKDIR) ? TrackdirToTrack(td_ret) : FindFirstTrack(tracks);
567 }
568 
570 {
571  const Train *last_veh = v->Last();
572 
573  /* get trackdirs of both ends */
574  Trackdir td = v->GetVehicleTrackdir();
575  Trackdir td_rev = ReverseTrackdir(last_veh->GetVehicleTrackdir());
576 
577  /* tiles where front and back are */
578  TileIndex tile = v->tile;
579  TileIndex tile_rev = last_veh->tile;
580 
581  int reverse_penalty = 0;
582 
583  if (v->track == TRACK_BIT_WORMHOLE) {
584  /* front in tunnel / on bridge */
585  DiagDirection dir_into_wormhole = GetTunnelBridgeDirection(tile);
586 
587  if (TrackdirToExitdir(td) == dir_into_wormhole) tile = GetOtherTunnelBridgeEnd(tile);
588  /* Now 'tile' is the tunnel entry/bridge ramp the train will reach when driving forward */
589 
590  /* Current position of the train in the wormhole */
591  TileIndex cur_tile = TileVirtXY(v->x_pos, v->y_pos);
592 
593  /* Add distance to drive in the wormhole as penalty for the forward path, i.e. bonus for the reverse path
594  * Note: Negative penalties are ok for the start tile. */
595  reverse_penalty -= DistanceManhattan(cur_tile, tile) * YAPF_TILE_LENGTH;
596  }
597 
598  if (last_veh->track == TRACK_BIT_WORMHOLE) {
599  /* back in tunnel / on bridge */
600  DiagDirection dir_into_wormhole = GetTunnelBridgeDirection(tile_rev);
601 
602  if (TrackdirToExitdir(td_rev) == dir_into_wormhole) tile_rev = GetOtherTunnelBridgeEnd(tile_rev);
603  /* Now 'tile_rev' is the tunnel entry/bridge ramp the train will reach when reversing */
604 
605  /* Current position of the last wagon in the wormhole */
606  TileIndex cur_tile = TileVirtXY(last_veh->x_pos, last_veh->y_pos);
607 
608  /* Add distance to drive in the wormhole as penalty for the revere path. */
609  reverse_penalty += DistanceManhattan(cur_tile, tile_rev) * YAPF_TILE_LENGTH;
610  }
611 
612  typedef bool (*PfnCheckReverseTrain)(const Train*, TileIndex, Trackdir, TileIndex, Trackdir, int);
613  PfnCheckReverseTrain pfnCheckReverseTrain = CYapfRail1::stCheckReverseTrain;
614 
615  /* check if non-default YAPF type needed */
617  pfnCheckReverseTrain = &CYapfRail2::stCheckReverseTrain; // Trackdir, forbid 90-deg
618  }
619 
620  /* slightly hackish: If the pathfinders finds a path, the cost of the first node is tested to distinguish between forward- and reverse-path. */
621  if (reverse_penalty == 0) reverse_penalty = 1;
622 
623  bool reverse = pfnCheckReverseTrain(v, tile, td, tile_rev, td_rev, reverse_penalty);
624 
625  return reverse;
626 }
627 
629 {
630  const Train *last_veh = v->Last();
631 
633  TileIndex last_tile = last_veh->tile;
634  Trackdir td_rev = ReverseTrackdir(last_veh->GetVehicleTrackdir());
635 
636  typedef FindDepotData (*PfnFindNearestDepotTwoWay)(const Train*, TileIndex, Trackdir, TileIndex, Trackdir, int, int);
637  PfnFindNearestDepotTwoWay pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail1::stFindNearestDepotTwoWay;
638 
639  /* check if non-default YAPF type needed */
641  pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail2::stFindNearestDepotTwoWay; // Trackdir, forbid 90-deg
642  }
643 
644  return pfnFindNearestDepotTwoWay(v, origin.tile, origin.trackdir, last_tile, td_rev, max_penalty, YAPF_INFINITE_PENALTY);
645 }
646 
647 bool YapfTrainFindNearestSafeTile(const Train *v, TileIndex tile, Trackdir td, bool override_railtype)
648 {
649  typedef bool (*PfnFindNearestSafeTile)(const Train*, TileIndex, Trackdir, bool);
650  PfnFindNearestSafeTile pfnFindNearestSafeTile = CYapfAnySafeTileRail1::stFindNearestSafeTile;
651 
652  /* check if non-default YAPF type needed */
654  pfnFindNearestSafeTile = &CYapfAnySafeTileRail2::stFindNearestSafeTile;
655  }
656 
657  return pfnFindNearestSafeTile(v, tile, td, override_railtype);
658 }
659 
662 
664 {
665  CSegmentCostCacheBase::NotifyTrackLayoutChange(tile, track);
666 }
PBSTileInfo::okay
bool okay
True if tile is a safe waiting position, false otherwise.
Definition: pbs.h:29
CYapfReserveTrack::m_res_dest_td
Trackdir m_res_dest_td
The reservation target trackdir.
Definition: yapf_rail.cpp:52
FindFirstTrack
Track FindFirstTrack(TrackBits tracks)
Returns first Track from TrackBits or INVALID_TRACK.
Definition: track_func.h:177
TileAdd
constexpr TileIndex TileAdd(TileIndex tile, TileIndexDiff offset)
Adds a given offset to a tile.
Definition: map_func.h:466
CYapfFollowAnyDepotRailT::Node
Types::NodeList::Titem Node
this will be our node type
Definition: yapf_rail.cpp:208
yapf_destrail.hpp
CYapfReserveTrack::Yapf
Tpf & Yapf()
to access inherited pathfinder
Definition: yapf_rail.cpp:45
CYapfFollowAnySafeTileRailT::Yapf
Tpf & Yapf()
to access inherited path finder
Definition: yapf_rail.cpp:304
FindDepotData
Helper container to find a depot.
Definition: pathfinder_type.h:38
PathfinderSettings::forbid_90_deg
bool forbid_90_deg
forbid trains to make 90 deg turns
Definition: settings_type.h:466
CYapfReserveTrack::ReserveRailStationPlatform
bool ReserveRailStationPlatform(TileIndex &tile, DiagDirection dir)
Reserve a railway platform.
Definition: yapf_rail.cpp:71
CYapfReserveTrack::Tpf
Types::Tpf Tpf
the pathfinder class (derived from THIS class)
Definition: yapf_rail.cpp:39
CYapfReserveTrack::m_res_node
Node * m_res_node
The reservation target node.
Definition: yapf_rail.cpp:53
YapfNotifyTrackLayoutChange
void YapfNotifyTrackLayoutChange(TileIndex tile, Track track)
Use this function to notify YAPF that track layout (or signal configuration) has change.
Definition: yapf_rail.cpp:663
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
yapf.hpp
FindDepotData::reverse
bool reverse
True if reversing is necessary for the train to get to this depot.
Definition: pathfinder_type.h:41
CYapfFollowRailT::Tpf
Types::Tpf Tpf
the pathfinder class (derived from THIS class)
Definition: yapf_rail.cpp:380
FollowTrainReservation
PBSTileInfo FollowTrainReservation(const Train *v, Vehicle **train_on_res)
Follow a train reservation to the last tile.
Definition: pbs.cpp:288
CYapfReserveTrack::FindSafePositionOnNode
void FindSafePositionOnNode(Node *node)
Check the node for a possible reservation target.
Definition: yapf_rail.cpp:143
yapf_node_rail.hpp
INVALID_TILE
constexpr TileIndex INVALID_TILE
The very nice invalid tile marker.
Definition: tile_type.h:95
SIGNAL_STATE_GREEN
@ SIGNAL_STATE_GREEN
The signal is green.
Definition: signal_type.h:44
IsCompatibleTrainStationTile
bool IsCompatibleTrainStationTile(Tile test_tile, Tile station_tile)
Check if a tile is a valid continuation to a railstation tile.
Definition: station_map.h:537
IsRailStationTile
bool IsRailStationTile(Tile t)
Is this tile a station tile and a rail station?
Definition: station_map.h:102
DiagDirection
DiagDirection
Enumeration for diagonal directions.
Definition: direction_type.h:73
CYapfCostRailT
Definition: yapf_costrail.hpp:17
StrongType::Typedef< uint32_t, struct TileIndexTag, StrongType::Compare, StrongType::Integer, StrongType::Compatible< int32_t >, StrongType::Compatible< int64_t > >
CYapfT
YAPF template that uses Ttypes template argument to determine all YAPF components (base classes) from...
Definition: yapf_common.hpp:183
FileHandle::Open
static std::optional< FileHandle > Open(const std::string &filename, const std::string &mode)
Open an RAII file handle if possible.
Definition: fileio.cpp:1170
CYapfFollowAnySafeTileRailT::PfFollowNode
void PfFollowNode(Node &old_node)
Called by YAPF to move from the given node to the next tile.
Definition: yapf_rail.cpp:315
CYapfFollowAnyDepotRailT
Definition: yapf_rail.cpp:203
CYapfFollowAnyDepotRailT::TransportTypeChar
char TransportTypeChar() const
return debug report character to identify the transportation type
Definition: yapf_rail.cpp:233
CYapfReserveTrack::m_res_fail_td
Trackdir m_res_fail_td
The trackdir where the reservation failed.
Definition: yapf_rail.cpp:55
Debug
#define Debug(category, level, format_string,...)
Ouptut a line of debugging information.
Definition: debug.h:37
TryReserveRailTrack
bool TryReserveRailTrack(TileIndex tile, Track t, bool trigger_stations)
Try to reserve a specific track on a tile.
Definition: pbs.cpp:80
Vehicle::x_pos
int32_t x_pos
x coordinate.
Definition: vehicle_base.h:304
CYapfFollowRailT::Key
Node::Key Key
key to hash tables
Definition: yapf_rail.cpp:383
CYapfFollowRailT::TransportTypeChar
char TransportTypeChar() const
return debug report character to identify the transportation type
Definition: yapf_rail.cpp:407
GameSettings::pf
PathfinderSettings pf
settings for all pathfinders
Definition: settings_type.h:600
DistanceManhattan
uint DistanceManhattan(TileIndex t0, TileIndex t1)
Gets the Manhattan distance between the two given tiles.
Definition: map.cpp:140
UnreserveRailTrack
void UnreserveRailTrack(TileIndex tile, Track t)
Lift the reservation of a specific track on a tile.
Definition: pbs.cpp:140
IsWaitingPositionFree
bool IsWaitingPositionFree(const Train *v, TileIndex tile, Trackdir trackdir, bool forbid_90deg)
Check if a safe position is free.
Definition: pbs.cpp:426
CYapfOriginTileTwoWayT
YAPF origin provider base class - used when there are two tile/trackdir origins.
Definition: yapf_common.hpp:55
CYapfReserveTrack::m_origin_tile
TileIndex m_origin_tile
Tile our reservation will originate from.
Definition: yapf_rail.cpp:56
Train::GetVehicleTrackdir
Trackdir GetVehicleTrackdir() const override
Get the tracks of the train vehicle.
Definition: train_cmd.cpp:4225
CYapfAnySafeTileRail2
Definition: yapf_rail.cpp:551
TrackBits
TrackBits
Allow incrementing of Track variables.
Definition: track_type.h:35
YAPF_TILE_LENGTH
static const int YAPF_TILE_LENGTH
Length (penalty) of one tile with YAPF.
Definition: pathfinder_type.h:16
Vehicle::tile
TileIndex tile
Current tile index.
Definition: vehicle_base.h:264
SIGNAL_STATE_RED
@ SIGNAL_STATE_RED
The signal is red.
Definition: signal_type.h:43
CYapfRail_TypesT
Definition: yapf_rail.cpp:528
PBSTileInfo
This struct contains information about the end of a reserved path.
Definition: pbs.h:26
CYapfFollowAnySafeTileRailT::Key
Node::Key Key
key to hash tables
Definition: yapf_rail.cpp:300
PBSTileInfo::trackdir
Trackdir trackdir
The reserved trackdir on the tile.
Definition: pbs.h:28
CYapfSegmentCostCacheGlobalT
CYapfSegmentCostCacheGlobalT - the yapf cost cache provider that adds the segment cost caching functi...
Definition: yapf_costcache.hpp:105
ReverseTrackdir
Trackdir ReverseTrackdir(Trackdir trackdir)
Maps a trackdir to the reverse trackdir.
Definition: track_func.h:247
DumpTarget
Class that represents the dump-into-string target.
Definition: dbg_helpers.h:95
_settings_game
GameSettings _settings_game
Game settings of a running game or the scenario editor.
Definition: settings.cpp:57
CYapfRail1
Definition: yapf_rail.cpp:544
CYapfAnyDepotRail2
Definition: yapf_rail.cpp:548
Train
'Train' is either a loco or a wagon.
Definition: train.h:89
CYapfReserveTrack::UnreserveSingleTrack
bool UnreserveSingleTrack(TileIndex tile, Trackdir td)
Unreserve a single track/platform.
Definition: yapf_rail.cpp:118
CYapfReserveTrack::SetReservationTarget
void SetReservationTarget(Node *node, TileIndex tile, Trackdir td)
Set the target to where the reservation should be extended.
Definition: yapf_rail.cpp:135
CYapfReserveTrack::TryReservePath
bool TryReservePath(PBSTileInfo *target, TileIndex origin)
Try to reserve the path till the reservation target.
Definition: yapf_rail.cpp: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
CYapfFollowAnySafeTileRailT::Tpf
Types::Tpf Tpf
the pathfinder class (derived from THIS class)
Definition: yapf_rail.cpp:297
TileIndexDiff
int32_t TileIndexDiff
An offset value between two tiles.
Definition: map_func.h:376
SetRailStationReservation
void SetRailStationReservation(Tile t, bool b)
Set the reservation state of the rail station.
Definition: station_map.h:564
YapfTrainFindNearestDepot
FindDepotData YapfTrainFindNearestDepot(const Train *v, int max_penalty)
Used when user sends train to the nearest depot or if train needs servicing using YAPF.
Definition: yapf_rail.cpp:628
CYapfFollowAnySafeTileRailT::TransportTypeChar
char TransportTypeChar() const
Return debug report character to identify the transportation type.
Definition: yapf_rail.cpp:324
CSegmentCostCacheBase::s_rail_change_counter
static int s_rail_change_counter
if any track changes, this counter is incremented - that will invalidate segment cost cache
Definition: yapf_costcache.hpp:46
yapf_cache.h
TRACK_BIT_WORMHOLE
@ TRACK_BIT_WORMHOLE
Bitflag for a wormhole (used for tunnels)
Definition: track_type.h:52
CYapfReserveTrack::m_res_fail_tile
TileIndex m_res_fail_tile
The tile where the reservation failed.
Definition: yapf_rail.cpp:54
TileOffsByDiagDir
TileIndexDiff TileOffsByDiagDir(DiagDirection dir)
Convert a DiagDirection to a TileIndexDiff.
Definition: map_func.h:565
CYapfFollowAnyDepotRailT::Tpf
Types::Tpf Tpf
the pathfinder class (derived from THIS class)
Definition: yapf_rail.cpp:206
CYapfAnySafeTileRail1
Definition: yapf_rail.cpp:550
CYapfFollowRailT::Yapf
Tpf & Yapf()
to access inherited path finder
Definition: yapf_rail.cpp:387
CYapfReserveTrack::m_res_dest
TileIndex m_res_dest
The reservation target tile.
Definition: yapf_rail.cpp:51
PBSTileInfo::tile
TileIndex tile
Tile the path ends, INVALID_TILE if no valid path was found.
Definition: pbs.h:27
Track
Track
These are used to specify a single track.
Definition: track_type.h:19
YapfTrainChooseTrack
Track YapfTrainChooseTrack(const Train *v, TileIndex tile, DiagDirection enterdir, TrackBits tracks, bool &path_found, bool reserve_track, PBSTileInfo *target, TileIndex *dest)
Finds the best path for given train using YAPF.
Definition: yapf_rail.cpp:554
Trackdir
Trackdir
Enumeration for tracks and directions.
Definition: track_type.h:67
CYapfRail2
Definition: yapf_rail.cpp:545
HasStationReservation
bool HasStationReservation(Tile t)
Get the reservation state of the rail station.
Definition: station_map.h:552
CYapfAnyDepotRail1
Definition: yapf_rail.cpp:547
MarkTileDirtyByTile
void MarkTileDirtyByTile(TileIndex tile, int bridge_level_offset, int tile_height_override)
Mark a tile given by its index dirty for repaint.
Definition: viewport.cpp:2054
DumpTarget::m_out
std::string m_out
the output string
Definition: dbg_helpers.h:118
CYapfReserveTrack
Definition: yapf_rail.cpp:36
CYapfFollowAnyDepotRailT::PfFollowNode
void PfFollowNode(Node &old_node)
Called by YAPF to move from the given node to the next tile.
Definition: yapf_rail.cpp:224
FindDepotData::tile
TileIndex tile
The tile of the depot.
Definition: pathfinder_type.h:39
YAPF_INFINITE_PENALTY
static const int YAPF_INFINITE_PENALTY
This penalty is the equivalent of "infinite", which means that paths that get this penalty will be ch...
Definition: pathfinder_type.h:27
CYapfReserveTrack::m_signals_set_to_red
std::vector< std::pair< TileIndex, Trackdir > > m_signals_set_to_red
List of signals turned red during a path reservation.
Definition: yapf_rail.cpp:58
TileIndex
StrongType::Typedef< uint32_t, struct TileIndexTag, StrongType::Compare, StrongType::Integer, StrongType::Compatible< int32_t >, StrongType::Compatible< int64_t > > TileIndex
The index/ID of a Tile.
Definition: tile_type.h:87
YapfTrainCheckReverse
bool YapfTrainCheckReverse(const Train *v)
Returns true if it is better to reverse the train before leaving station using YAPF.
Definition: yapf_rail.cpp:569
GetOtherTunnelBridgeEnd
TileIndex GetOtherTunnelBridgeEnd(Tile t)
Determines type of the wormhole and returns its other end.
Definition: tunnelbridge_map.h:78
Vehicle::y_pos
int32_t y_pos
y coordinate.
Definition: vehicle_base.h:305
CYapfFollowRailT
Definition: yapf_rail.cpp:377
TriggerStationRandomisation
void TriggerStationRandomisation(Station *st, TileIndex trigger_tile, StationRandomTrigger trigger, CargoID cargo_type)
Trigger station randomisation.
Definition: newgrf_station.cpp:919
SRT_PATH_RESERVATION
@ SRT_PATH_RESERVATION
Trigger platform when train reserves path.
Definition: newgrf_station.h:111
CYapfFollowAnySafeTileRailT
Definition: yapf_rail.cpp:294
CYapfFollowRailT::PfFollowNode
void PfFollowNode(Node &old_node)
Called by YAPF to move from the given node to the next tile.
Definition: yapf_rail.cpp:398
TileVirtXY
static debug_inline TileIndex TileVirtXY(uint x, uint y)
Get a tile from the virtual XY-coordinate.
Definition: map_func.h:416
CYapfFollowAnySafeTileRailT::Node
Types::NodeList::Titem Node
this will be our node type
Definition: yapf_rail.cpp:299
CYapfBaseT
CYapfBaseT - A-star type path finder base class.
Definition: yapf_base.hpp:47
CYapfFollowRailT::Node
Types::NodeList::Titem Node
this will be our node type
Definition: yapf_rail.cpp:382
LinkGraph::BaseNode
Node of the link graph.
Definition: linkgraph.h:90
HasPbsSignalOnTrackdir
bool HasPbsSignalOnTrackdir(Tile tile, Trackdir td)
Is a pbs signal present along the trackdir?
Definition: rail_map.h:463
CYapfFollowAnyDepotRailT::Key
Node::Key Key
key to hash tables
Definition: yapf_rail.cpp:209
SpecializedVehicle::Last
T * Last()
Get the last vehicle in the chain.
Definition: vehicle_base.h:1118
INVALID_TRACK
@ INVALID_TRACK
Flag for an invalid track.
Definition: track_type.h:28
CYapfFollowAnyDepotRailT::Yapf
Tpf & Yapf()
to access inherited path finder
Definition: yapf_rail.cpp:213
SetSignalStateByTrackdir
void SetSignalStateByTrackdir(Tile tile, Trackdir trackdir, SignalState state)
Sets the state of the signal along the given trackdir.
Definition: rail_map.h:449
INVALID_TRACKDIR
@ INVALID_TRACKDIR
Flag for an invalid trackdir.
Definition: track_type.h:86
CYapfReserveTrack::Node
Types::NodeList::Titem Node
this will be our node type
Definition: yapf_rail.cpp:41
YapfTrainFindNearestSafeTile
bool YapfTrainFindNearestSafeTile(const Train *v, TileIndex tile, Trackdir td, bool override_railtype)
Try to extend the reserved path of a train to the nearest safe tile using YAPF.
Definition: yapf_rail.cpp:647
yapf_costrail.hpp
CYapfReserveTrack::ReserveSingleTrack
bool ReserveSingleTrack(TileIndex tile, Trackdir td)
Try to reserve a single track/platform.
Definition: yapf_rail.cpp:89
GetTunnelBridgeDirection
DiagDirection GetTunnelBridgeDirection(Tile t)
Get the direction pointing to the other end.
Definition: tunnelbridge_map.h:26