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/>.
8 /** @file yapf_costcache.hpp Caching of segment costs. */
10 #ifndef YAPF_COSTCACHE_HPP
11 #define YAPF_COSTCACHE_HPP
13 #include "../../timer/timer_game_calendar.h"
16 * CYapfSegmentCostCacheNoneT - the formal only yapf cost cache provider that implements
17 * PfNodeCacheFetch(). Used when nodes don't have CachedData
18 * defined (they don't count with any segment cost caching).
20 template <class Types
>
21 class CYapfSegmentCostCacheNoneT
24 typedef typename
Types::Tpf Tpf
; ///< the pathfinder class (derived from THIS class)
25 typedef typename
Types::NodeList::Titem Node
; ///< this will be our node type
28 * Called by YAPF to attach cached or local segment cost data to the given node.
29 * @return true if globally cached data were used or false if local data was used
31 inline bool PfNodeCacheFetch(Node
&)
38 * Base class for segment cost cache providers. Contains global counter
39 * of track layout changes and static notification function called whenever
40 * the track layout changes. It is implemented as base class because it needs
41 * to be shared between all rail YAPF types (one shared counter, one notification
44 struct CSegmentCostCacheBase
46 static int s_rail_change_counter
;
48 static void NotifyTrackLayoutChange(TileIndex
, Track
)
50 s_rail_change_counter
++;
56 * CSegmentCostCacheT - template class providing hash-map and storage (heap)
57 * of Tsegment structures. Each rail node contains pointer to the segment
58 * that contains cached (or non-cached) segment cost information. Nodes can
59 * differ by key type, but they use the same segment type. Segment key should
60 * be always the same (TileIndex + DiagDirection) that represent the beginning
61 * of the segment (origin tile and exit-dir from this tile).
62 * Different CYapfCachedCostT types can share the same type of CSegmentCostCacheT.
63 * Look at CYapfRailSegment (yapf_node_rail.hpp) for the segment example
65 template <class Tsegment
>
66 struct CSegmentCostCacheT
: public CSegmentCostCacheBase
{
67 static const int C_HASH_BITS
= 14;
69 typedef CHashTableT
<Tsegment
, C_HASH_BITS
> HashTable
;
70 using Heap
= std::deque
<Tsegment
>;
71 typedef typename
Tsegment::Key Key
; ///< key to hash table
76 inline CSegmentCostCacheT() {}
78 /** flush (clear) the cache */
85 inline Tsegment
&Get(Key
&key
, bool *found
)
87 Tsegment
*item
= m_map
.Find(key
);
88 if (item
== nullptr) {
90 item
= &m_heap
.emplace_back(key
);
100 * CYapfSegmentCostCacheGlobalT - the yapf cost cache provider that adds the segment cost
101 * caching functionality to yapf. Using this class as base of your will provide the global
102 * segment cost caching services for your Nodes.
104 template <class Types
>
105 class CYapfSegmentCostCacheGlobalT
{
107 typedef typename
Types::Tpf Tpf
; ///< the pathfinder class (derived from THIS class)
108 typedef typename
Types::NodeList::Titem Node
; ///< this will be our node type
109 typedef typename
Node::Key Key
; ///< key to hash tables
110 typedef typename
Node::CachedData CachedData
;
111 typedef typename
CachedData::Key CacheKey
;
112 typedef CSegmentCostCacheT
<CachedData
> Cache
;
113 using LocalCache
= std::deque
<CachedData
>;
116 Cache
&m_global_cache
;
117 LocalCache m_local_cache
;
119 inline CYapfSegmentCostCacheGlobalT() : m_global_cache(stGetGlobalCache()) {};
121 /** to access inherited path finder */
124 return *static_cast<Tpf
*>(this);
127 inline static Cache
&stGetGlobalCache()
129 static int last_rail_change_counter
= 0;
132 /* delete the cache sometimes... */
133 if (last_rail_change_counter
!= Cache::s_rail_change_counter
) {
134 last_rail_change_counter
= Cache::s_rail_change_counter
;
142 * Called by YAPF to attach cached or local segment cost data to the given node.
143 * @return true if globally cached data were used or false if local data was used
145 inline bool PfNodeCacheFetch(Node
&n
)
147 CacheKey
key(n
.GetKey());
149 if (!Yapf().CanUseGlobalCache(n
)) {
150 Yapf().ConnectNodeToCachedData(n
, m_local_cache
.emplace_back(key
));
155 CachedData
&item
= m_global_cache
.Get(key
, &found
);
156 Yapf().ConnectNodeToCachedData(n
, item
);
161 #endif /* YAPF_COSTCACHE_HPP */