Changeset - r7084:ae49552087c9
[Not reviewed]
master
0 2 0
KUDr - 17 years ago 2007-06-26 22:29:52
kudr@openttd.org
(svn r10349) -Fix [FS#941, YAPF, r10301]: tile/trackdir must be used as node key and also as segment key in the cache (SmatZ).
In the past it was possible to use tile/exitdir as the key because segments beginning on the same tile/exitdir were incorrectly considered the same. What I still don't understand is why this bug happened only on 64 bit systems (linux, Win64).
2 files changed with 17 insertions and 24 deletions:
0 comments (0 inline, 0 general)
src/yapf/yapf_node_rail.hpp
Show inline comments
 
@@ -11,14 +11,13 @@ struct CYapfRailSegmentKey
 
	uint32    m_value;
 

	
 
	FORCEINLINE CYapfRailSegmentKey(const CYapfRailSegmentKey& src) : m_value(src.m_value) {}
 
	FORCEINLINE CYapfRailSegmentKey(const CYapfNodeKeyExitDir& node_key) {Set(node_key);}
 
	FORCEINLINE CYapfRailSegmentKey(const CYapfNodeKeyTrackDir& node_key) {Set(node_key);}
 

	
 
	FORCEINLINE void Set(const CYapfRailSegmentKey& src) {m_value = src.m_value;}
 
	FORCEINLINE void Set(const CYapfNodeKeyExitDir& node_key) {m_value = (((int)node_key.m_tile) << 2) | node_key.m_exitdir;}
 
	FORCEINLINE void Set(const CYapfNodeKeyTrackDir& node_key) {m_value = (((int)node_key.m_tile) << 3) | node_key.m_td;}
 

	
 
	FORCEINLINE int32 CalcHash() const {return m_value;}
 
	FORCEINLINE TileIndex GetTile() const {return (TileIndex)(m_value >> 2);}
 
	FORCEINLINE DiagDirection GetExitDir() const {return (DiagDirection)(m_value & 3);}
 
	FORCEINLINE TileIndex GetTile() const {return (TileIndex)(m_value >> 3);}
 
	FORCEINLINE bool operator == (const CYapfRailSegmentKey& other) const {return m_value == other.m_value;}
 
};
 

	
 
@@ -103,7 +102,6 @@ struct CYapfRailSegment
 

	
 
	FORCEINLINE const Key& GetKey() const {return m_key;}
 
	FORCEINLINE TileIndex GetTile() const {return m_key.GetTile();}
 
	FORCEINLINE DiagDirection GetExitDir() const {return m_key.GetExitDir();}
 
	FORCEINLINE CYapfRailSegment* GetHashNext() {return m_hash_next;}
 
	FORCEINLINE void SetHashNext(CYapfRailSegment* next) {m_hash_next = next;}
 
};
src/yapf/yapf_rail.cpp
Show inline comments
 
@@ -194,25 +194,22 @@ struct CYapfRail_TypesT
 
};
 

	
 
struct CYapfRail1         : CYapfT<CYapfRail_TypesT<CYapfRail1        , CFollowTrackRail    , CRailNodeListTrackDir, CYapfDestinationTileOrStationRailT, CYapfFollowRailT> > {};
 
struct CYapfRail2         : CYapfT<CYapfRail_TypesT<CYapfRail2        , CFollowTrackRail    , CRailNodeListExitDir , CYapfDestinationTileOrStationRailT, CYapfFollowRailT> > {};
 
struct CYapfRail3         : CYapfT<CYapfRail_TypesT<CYapfRail3        , CFollowTrackRailNo90, CRailNodeListTrackDir, CYapfDestinationTileOrStationRailT, CYapfFollowRailT> > {};
 
struct CYapfRail2         : CYapfT<CYapfRail_TypesT<CYapfRail2        , CFollowTrackRailNo90, CRailNodeListTrackDir, CYapfDestinationTileOrStationRailT, CYapfFollowRailT> > {};
 

	
 
struct CYapfAnyDepotRail1 : CYapfT<CYapfRail_TypesT<CYapfAnyDepotRail1, CFollowTrackRail    , CRailNodeListTrackDir, CYapfDestinationAnyDepotRailT     , CYapfFollowAnyDepotRailT> > {};
 
struct CYapfAnyDepotRail2 : CYapfT<CYapfRail_TypesT<CYapfAnyDepotRail2, CFollowTrackRail    , CRailNodeListExitDir , CYapfDestinationAnyDepotRailT     , CYapfFollowAnyDepotRailT> > {};
 
struct CYapfAnyDepotRail3 : CYapfT<CYapfRail_TypesT<CYapfAnyDepotRail3, CFollowTrackRailNo90, CRailNodeListTrackDir, CYapfDestinationAnyDepotRailT     , CYapfFollowAnyDepotRailT> > {};
 
struct CYapfAnyDepotRail2 : CYapfT<CYapfRail_TypesT<CYapfAnyDepotRail2, CFollowTrackRailNo90, CRailNodeListTrackDir, CYapfDestinationAnyDepotRailT     , CYapfFollowAnyDepotRailT> > {};
 

	
 

	
 
Trackdir YapfChooseRailTrack(Vehicle *v, TileIndex tile, DiagDirection enterdir, TrackBits tracks, bool *path_not_found)
 
{
 
	// default is YAPF type 2
 
	typedef Trackdir (*PfnChooseRailTrack)(Vehicle*, TileIndex, DiagDirection, TrackBits, bool*);
 
	PfnChooseRailTrack pfnChooseRailTrack = &CYapfRail2::stChooseRailTrack;
 
	PfnChooseRailTrack pfnChooseRailTrack = &CYapfRail1::stChooseRailTrack;
 

	
 
	// check if non-default YAPF type needed
 
	if (_patches.forbid_90_deg)
 
		pfnChooseRailTrack = &CYapfRail3::stChooseRailTrack; // Trackdir, forbid 90-deg
 
	else if (_patches.yapf.disable_node_optimization)
 
		pfnChooseRailTrack = &CYapfRail1::stChooseRailTrack; // Trackdir, allow 90-deg
 
	if (_patches.forbid_90_deg) {
 
		pfnChooseRailTrack = &CYapfRail2::stChooseRailTrack; // Trackdir, forbid 90-deg
 
	}
 

	
 
	Trackdir td_ret = pfnChooseRailTrack(v, tile, enterdir, tracks, path_not_found);
 

	
 
@@ -233,13 +230,12 @@ bool YapfCheckReverseTrain(Vehicle* v)
 

	
 

	
 
	typedef bool (*PfnCheckReverseTrain)(Vehicle*, TileIndex, Trackdir, TileIndex, Trackdir);
 
	PfnCheckReverseTrain pfnCheckReverseTrain = CYapfRail2::stCheckReverseTrain;
 
	PfnCheckReverseTrain pfnCheckReverseTrain = CYapfRail1::stCheckReverseTrain;
 

	
 
	// check if non-default YAPF type needed
 
	if (_patches.forbid_90_deg)
 
		pfnCheckReverseTrain = &CYapfRail3::stCheckReverseTrain; // Trackdir, forbid 90-deg
 
	else if (_patches.yapf.disable_node_optimization)
 
		pfnCheckReverseTrain = &CYapfRail1::stCheckReverseTrain; // Trackdir, allow 90-deg
 
	if (_patches.forbid_90_deg) {
 
		pfnCheckReverseTrain = &CYapfRail2::stCheckReverseTrain; // Trackdir, forbid 90-deg
 
	}
 

	
 
	bool reverse = pfnCheckReverseTrain(v, tile, td, last_veh->tile, td_rev);
 

	
 
@@ -261,13 +257,12 @@ bool YapfFindNearestRailDepotTwoWay(Vehi
 
	Trackdir td_rev = ReverseTrackdir(GetVehicleTrackdir(last_veh));
 

	
 
	typedef bool (*PfnFindNearestDepotTwoWay)(Vehicle*, TileIndex, Trackdir, TileIndex, Trackdir, int, int, TileIndex*, bool*);
 
	PfnFindNearestDepotTwoWay pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail2::stFindNearestDepotTwoWay;
 
	PfnFindNearestDepotTwoWay pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail1::stFindNearestDepotTwoWay;
 

	
 
	// check if non-default YAPF type needed
 
	if (_patches.forbid_90_deg)
 
		pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail3::stFindNearestDepotTwoWay; // Trackdir, forbid 90-deg
 
	else if (_patches.yapf.disable_node_optimization)
 
		pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail1::stFindNearestDepotTwoWay; // Trackdir, allow 90-deg
 
	if (_patches.forbid_90_deg) {
 
		pfnFindNearestDepotTwoWay = &CYapfAnyDepotRail2::stFindNearestDepotTwoWay; // Trackdir, forbid 90-deg
 
	}
 

	
 
	bool ret = pfnFindNearestDepotTwoWay(v, tile, td, last_tile, td_rev, max_distance, reverse_penalty, depot_tile, reversed);
 
	return ret;
0 comments (0 inline, 0 general)