2005-07-24 14:12:37 +00:00
/* $Id$ */
2005-01-31 11:23:10 +00:00
# include "stdafx.h"
2005-06-02 19:30:21 +00:00
# include "openttd.h"
2005-02-05 15:58:59 +00:00
# include "debug.h"
2005-07-22 07:02:20 +00:00
# include "functions.h"
2005-01-31 11:23:10 +00:00
# include "npf.h"
# include "aystar.h"
# include "macros.h"
# include "pathfind.h"
# include "station.h"
# include "tile.h"
2005-02-06 10:18:47 +00:00
# include "depot.h"
2005-01-31 11:23:10 +00:00
2005-06-25 19:08:58 +00:00
static AyStar _npf_aystar ;
2005-01-31 11:23:10 +00:00
/* The cost of each trackdir. A diagonal piece is the full NPF_TILE_LENGTH,
* the shorter piece is sqrt ( 2 ) / 2 * NPF_TILE_LENGTH = ~ 0.7071
*/
2005-04-11 19:14:48 +00:00
# define NPF_STRAIGHT_LENGTH (uint)(NPF_TILE_LENGTH * STRAIGHT_TRACK_LENGTH)
2005-06-17 17:11:05 +00:00
static const uint _trackdir_length [ TRACKDIR_END ] = {
2005-03-08 22:21:20 +00:00
NPF_TILE_LENGTH , NPF_TILE_LENGTH , NPF_STRAIGHT_LENGTH , NPF_STRAIGHT_LENGTH , NPF_STRAIGHT_LENGTH , NPF_STRAIGHT_LENGTH ,
0 , 0 ,
NPF_TILE_LENGTH , NPF_TILE_LENGTH , NPF_STRAIGHT_LENGTH , NPF_STRAIGHT_LENGTH , NPF_STRAIGHT_LENGTH , NPF_STRAIGHT_LENGTH
2005-01-31 11:23:10 +00:00
} ;
2005-09-09 23:14:38 +00:00
/**
* Calculates the minimum distance traveled to get from t0 to t1 when only
* using tracks ( ie , only making 45 degree turns ) . Returns the distance in the
* NPF scale , ie the number of full tiles multiplied by NPF_TILE_LENGTH to
* prevent rounding .
*/
2005-09-11 15:01:00 +00:00
static uint NPFDistanceTrack ( TileIndex t0 , TileIndex t1 )
2005-09-09 23:14:38 +00:00
{
const uint dx = abs ( TileX ( t0 ) - TileX ( t1 ) ) ;
const uint dy = abs ( TileY ( t0 ) - TileY ( t1 ) ) ;
const uint straightTracks = 2 * min ( dx , dy ) ; /* The number of straight (not full length) tracks */
/* OPTIMISATION:
* Original : diagTracks = max ( dx , dy ) - min ( dx , dy ) ;
* Proof :
* ( dx + dy ) - straightTracks = = ( min + max ) - straightTracks = min + max - 2 * min = max - min */
const uint diagTracks = dx + dy - straightTracks ; /* The number of diagonal (full tile length) tracks. */
/* Don't factor out NPF_TILE_LENGTH below, this will round values and lose
* precision */
return diagTracks * NPF_TILE_LENGTH + straightTracks * NPF_TILE_LENGTH * STRAIGHT_TRACK_LENGTH ;
}
2005-07-04 14:58:55 +00:00
/**
* Check if a rail track is the end of the line . Will also consider 1 - way signals to be the end of a line .
* @ param tile The tile on which the current track is .
2005-07-09 11:39:35 +00:00
* @ param trackdir The ( track ) direction in which you want to look .
* @ param enginetype The type of the engine for which we are checking this .
2005-07-04 14:58:55 +00:00
*/
2005-09-11 15:01:00 +00:00
static bool IsEndOfLine ( TileIndex tile , Trackdir trackdir , RailType enginetype )
2005-07-04 14:58:55 +00:00
{
byte exitdir = TrackdirToExitdir ( trackdir ) ;
TileIndex dst_tile ;
uint32 ts ;
2005-09-09 23:14:38 +00:00
/* Can always go into a tunnel */
2005-10-05 07:20:26 +00:00
if ( IsTileType ( tile , MP_TUNNELBRIDGE ) & & GB ( _m [ tile ] . m5 , 4 , 4 ) = = 0 & &
GB ( _m [ tile ] . m5 , 0 , 2 ) = = exitdir ) {
2005-07-04 14:58:55 +00:00
return false ;
2005-10-05 07:20:26 +00:00
}
2005-07-04 14:58:55 +00:00
2005-09-09 23:14:38 +00:00
/* Cannot go through the back of a depot */
2005-07-04 15:25:06 +00:00
if ( IsTileDepotType ( tile , TRANSPORT_RAIL ) & & ( exitdir ! = GetDepotDirection ( tile , TRANSPORT_RAIL ) ) )
return true ;
2005-07-04 14:58:55 +00:00
/* Calculate next tile */
dst_tile = tile + TileOffsByDir ( exitdir ) ;
// determine the track status on the next tile.
ts = GetTileTrackStatus ( dst_tile , TRANSPORT_RAIL ) & TrackdirReachesTrackdirs ( trackdir ) ;
// when none of the trackdir bits are set, we cant enter the new tile
if ( ( ts & TRACKDIR_BIT_MASK ) = = 0 )
return true ;
{
2005-07-04 15:25:06 +00:00
byte dst_type = GetTileRailType ( dst_tile , exitdir ) ;
2005-07-09 11:39:35 +00:00
if ( ! IsCompatibleRail ( enginetype , dst_type ) )
2005-07-04 14:58:55 +00:00
return true ;
if ( GetTileOwner ( tile ) ! = GetTileOwner ( dst_tile ) )
return true ;
2005-09-09 23:14:38 +00:00
/* Prevent us from entering a depot from behind */
2005-07-04 15:25:06 +00:00
if ( IsTileDepotType ( dst_tile , TRANSPORT_RAIL ) & & ( exitdir ! = ReverseDiagdir ( GetDepotDirection ( dst_tile , TRANSPORT_RAIL ) ) ) )
2005-07-04 14:58:55 +00:00
return true ;
2005-09-09 23:14:38 +00:00
/* Prevent us from falling off a slope into a tunnel exit */
2005-10-05 07:20:26 +00:00
if ( IsTileType ( dst_tile , MP_TUNNELBRIDGE ) & &
GB ( _m [ dst_tile ] . m5 , 4 , 4 ) = = 0 & &
( DiagDirection ) GB ( _m [ dst_tile ] . m5 , 0 , 2 ) = = ReverseDiagdir ( exitdir ) ) {
return true ;
}
2005-09-09 23:14:38 +00:00
2005-07-04 14:58:55 +00:00
/* Check for oneway signal against us */
if ( IsTileType ( dst_tile , MP_RAILWAY ) & & GetRailTileType ( dst_tile ) = = RAIL_TYPE_SIGNALS ) {
if ( HasSignalOnTrackdir ( dst_tile , ReverseTrackdir ( FindFirstBit2x64 ( ts ) ) ) & & ! HasSignalOnTrackdir ( dst_tile , FindFirstBit2x64 ( ts ) ) )
// if one way signal not pointing towards us, stop going in this direction.
return true ;
}
return false ;
}
2005-07-23 19:48:24 +00:00
}
2005-07-04 14:58:55 +00:00
2005-07-15 21:28:26 +00:00
#if 0
2005-06-25 19:08:58 +00:00
static uint NTPHash ( uint key1 , uint key2 )
2005-01-31 11:23:10 +00:00
{
2005-04-07 19:19:16 +00:00
/* This function uses the old hash, which is fixed on 10 bits (1024 buckets) */
2005-01-31 11:23:10 +00:00
return PATHFIND_HASH_TILE ( key1 ) ;
}
2005-07-15 21:28:26 +00:00
# endif
2005-01-31 11:23:10 +00:00
2005-06-17 17:11:05 +00:00
/**
* Calculates a hash value for use in the NPF .
* @ param key1 The TileIndex of the tile to hash
* @ param key1 The Trackdir of the track on the tile .
*
* @ todo Think of a better hash .
*/
2005-06-25 19:08:58 +00:00
static uint NPFHash ( uint key1 , uint key2 )
2005-04-07 19:19:16 +00:00
{
/* TODO: think of a better hash? */
uint part1 = TileX ( key1 ) & NPF_HASH_HALFMASK ;
uint part2 = TileY ( key1 ) & NPF_HASH_HALFMASK ;
2005-06-17 17:11:05 +00:00
assert ( IsValidTrackdir ( key2 ) ) ;
assert ( IsValidTile ( key1 ) ) ;
return ( ( ( ( part1 < < NPF_HASH_HALFBITS ) | part2 ) ) + ( NPF_HASH_SIZE * key2 / TRACKDIR_END ) ) % NPF_HASH_SIZE ;
2005-04-07 19:19:16 +00:00
}
2005-06-25 19:08:58 +00:00
static int32 NPFCalcZero ( AyStar * as , AyStarNode * current , OpenListNode * parent )
{
2005-01-31 11:23:10 +00:00
return 0 ;
}
2005-03-07 23:28:27 +00:00
/* Calcs the tile of given station that is closest to a given tile
* for this we assume the station is a rectangle ,
* as defined by its top tile ( st - > train_tile ) and its width / height ( st - > trainst_w , st - > trainst_h )
*/
2005-06-25 19:08:58 +00:00
static TileIndex CalcClosestStationTile ( StationID station , TileIndex tile )
{
2005-03-07 23:28:27 +00:00
const Station * st = GetStation ( station ) ;
2005-06-25 19:08:58 +00:00
uint minx = TileX ( st - > train_tile ) ; // topmost corner of station
uint miny = TileY ( st - > train_tile ) ;
uint maxx = minx + st - > trainst_w - 1 ; // lowermost corner of station
uint maxy = miny + st - > trainst_h - 1 ;
uint x ;
uint y ;
2005-03-07 23:28:27 +00:00
// we are going the aim for the x coordinate of the closest corner
// but if we are between those coordinates, we will aim for our own x coordinate
2005-06-25 19:08:58 +00:00
x = clamp ( TileX ( tile ) , minx , maxx ) ;
2005-03-07 23:28:27 +00:00
// same for y coordinate, see above comment
2005-06-25 19:08:58 +00:00
y = clamp ( TileY ( tile ) , miny , maxy ) ;
2005-03-07 23:28:27 +00:00
// return the tile of our target coordinates
2005-06-25 19:08:58 +00:00
return TileXY ( x , y ) ;
2005-07-23 19:48:24 +00:00
}
2005-03-07 23:28:27 +00:00
2005-07-04 14:58:55 +00:00
/* On PBS pathfinding runs, this is called before pathfinding ends (BeforeExit aystar callback), and will
* reserve the appropriate tracks , if needed . */
2005-09-11 15:01:00 +00:00
static void NPFReservePBSPath ( AyStar * as )
2005-07-04 14:58:55 +00:00
{
NPFFoundTargetData * ftd = ( NPFFoundTargetData * ) as - > user_path ;
bool eol_end = false ;
if ( ftd - > best_trackdir = = 0xFF )
return ;
2005-07-09 11:39:35 +00:00
if ( ! NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_EXIT ) & & IsEndOfLine ( ftd - > node . tile , ftd - > node . direction , as - > user_data [ NPF_RAILTYPE ] ) & & ! NPFGetFlag ( & ftd - > node , NPF_FLAG_SEEN_SIGNAL ) ) {
2005-07-04 14:58:55 +00:00
/* The path ends in an end of line, we'll need to reserve a path.
* We treat and end of line as a red exit signal */
eol_end = true ;
NPFSetFlag ( & ftd - > node , NPF_FLAG_PBS_EXIT , true ) ;
if ( ! NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_TARGET_SEEN ) )
NPFSetFlag ( & ftd - > node , NPF_FLAG_PBS_RED , true ) ;
}
if ( ! NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_CHOICE ) ) {
/* there have been no choices to make on our path, we dont care if our end signal is red */
NPFSetFlag ( & ftd - > node , NPF_FLAG_PBS_RED , false ) ;
}
if ( NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_EXIT ) & & // we passed an exit signal
! NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_BLOCKED ) & & // we didnt encounter reserver tracks
( ( as - > user_data [ NPF_PBS_MODE ] ! = PBS_MODE_GREEN ) | | ( ! NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_RED ) ) ) ) { // our mode permits having a red exit signal, or the signal is green
PathNode parent ;
PathNode * curr ;
PathNode * prev ;
TileIndex start = INVALID_TILE ;
byte trackdir = 0 ;
parent . node = ftd - > node ;
parent . parent = & ftd - > path ;
curr = & parent ;
prev = NULL ;
do {
if ( ! NPFGetFlag ( & curr - > node , NPF_FLAG_PBS_EXIT ) | | eol_end ) {
/* check for already reserved track on this path, if they clash with what we
currently trying to reserve , we have a self - crossing path : - ( */
if ( ( PBSTileUnavail ( curr - > node . tile ) & ( 1 < < curr - > node . direction ) )
& & ! ( PBSTileReserved ( curr - > node . tile ) & ( 1 < < ( curr - > node . direction & 7 ) ) )
& & ( start ! = INVALID_TILE ) ) {
/* It's actually quite bad if this happens, it means the pathfinder
* found a path that is intersecting with itself , which is a very bad
* thing in a pbs block . Also there is not much we can do about it at
* this point . . . .
* BUT , you have to have a pretty fucked up junction layout for this to happen ,
* so we ' ll just stop this train , the user will eventually notice , so he can fix it .
*/
2005-07-17 20:09:02 +00:00
PBSClearPath ( start , trackdir , curr - > node . tile , curr - > node . direction ) ;
2005-07-04 14:58:55 +00:00
NPFSetFlag ( & ftd - > node , NPF_FLAG_PBS_BLOCKED , true ) ;
DEBUG ( pbs , 1 ) ( " PBS: Self-crossing path!!! " ) ;
return ;
} ;
2005-07-17 20:09:02 +00:00
PBSReserveTrack ( curr - > node . tile , TrackdirToTrack ( curr - > node . direction ) ) ;
/* we want to reserve the last tile (with the signal) on the path too
also remember this tile , cause its the end of the path ( where we exit the block ) */
if ( start = = INVALID_TILE ) {
if ( prev ! = NULL ) {
PBSReserveTrack ( prev - > node . tile , TrackdirToTrack ( prev - > node . direction ) ) ;
start = prev - > node . tile ;
trackdir = ReverseTrackdir ( prev - > node . direction ) ;
} else {
start = curr - > node . tile ;
trackdir = curr - > node . direction ;
}
2005-07-04 14:58:55 +00:00
}
}
prev = curr ;
curr = curr - > parent ;
} while ( curr ! = NULL ) ;
2005-07-17 20:09:02 +00:00
// we remember the tile/track where this path leaves the pbs junction
ftd - > node . tile = start ;
ftd - > node . direction = trackdir ;
2005-07-04 14:58:55 +00:00
}
}
2005-04-11 19:14:48 +00:00
/* Calcs the heuristic to the target station or tile. For train stations, it
* takes into account the direction of approach .
2005-01-31 11:23:10 +00:00
*/
2005-06-25 19:08:58 +00:00
static int32 NPFCalcStationOrTileHeuristic ( AyStar * as , AyStarNode * current , OpenListNode * parent )
{
2005-01-31 11:23:10 +00:00
NPFFindStationOrTileData * fstd = ( NPFFindStationOrTileData * ) as - > user_target ;
NPFFoundTargetData * ftd = ( NPFFoundTargetData * ) as - > user_path ;
TileIndex from = current - > tile ;
TileIndex to = fstd - > dest_coords ;
2005-03-08 00:18:31 +00:00
uint dist ;
2005-03-07 23:28:27 +00:00
// for train-stations, we are going to aim for the closest station tile
if ( ( as - > user_data [ NPF_TYPE ] = = TRANSPORT_RAIL ) & & ( fstd - > station_index ! = - 1 ) )
2005-03-08 00:18:31 +00:00
to = CalcClosestStationTile ( fstd - > station_index , from ) ;
2005-03-07 23:28:27 +00:00
2005-04-11 19:14:48 +00:00
if ( as - > user_data [ NPF_TYPE ] = = TRANSPORT_ROAD )
/* Since roads only have diagonal pieces, we use manhattan distance here */
dist = DistanceManhattan ( from , to ) * NPF_TILE_LENGTH ;
else
/* Ships and trains can also go diagonal, so the minimum distance is shorter */
2005-09-09 23:14:38 +00:00
dist = NPFDistanceTrack ( from , to ) ;
2005-01-31 11:23:10 +00:00
2005-07-04 14:58:55 +00:00
DEBUG ( npf , 4 ) ( " Calculating H for: (%d, %d). Result: %d " , TileX ( current - > tile ) , TileY ( current - > tile ) , dist ) ;
/* for pbs runs, we ignore tiles inside the pbs block for the tracking
of the ' closest ' tile */
if ( ( as - > user_data [ NPF_PBS_MODE ] ! = PBS_MODE_NONE )
& & ( ! NPFGetFlag ( current , NPF_FLAG_SEEN_SIGNAL ) )
2005-07-09 11:39:35 +00:00
& & ( ! IsEndOfLine ( current - > tile , current - > direction , as - > user_data [ NPF_RAILTYPE ] ) ) )
2005-07-04 14:58:55 +00:00
return dist ;
if ( ( dist < ftd - > best_bird_dist ) | |
/* for pbs runs, prefer tiles that pass a green exit signal to the pbs blocks */
( ( as - > user_data [ NPF_PBS_MODE ] ! = PBS_MODE_NONE ) & & ! NPFGetFlag ( current , NPF_FLAG_PBS_RED ) & & NPFGetFlag ( & ftd - > node , NPF_FLAG_PBS_RED ) )
) {
2005-01-31 11:23:10 +00:00
ftd - > best_bird_dist = dist ;
ftd - > best_trackdir = current - > user_data [ NPF_TRACKDIR_CHOICE ] ;
2005-07-04 14:58:55 +00:00
ftd - > path = parent - > path ;
ftd - > node = * current ;
2005-01-31 11:23:10 +00:00
}
return dist ;
}
/* Fills AyStarNode.user_data[NPF_TRACKDIRCHOICE] with the chosen direction to
* get here , either getting it from the current choice or from the parent ' s
* choice */
2005-06-25 19:08:58 +00:00
static void NPFFillTrackdirChoice ( AyStarNode * current , OpenListNode * parent )
2005-01-31 11:23:10 +00:00
{
if ( parent - > path . parent = = NULL ) {
2005-06-17 17:11:05 +00:00
Trackdir trackdir = ( Trackdir ) current - > direction ;
2005-01-31 11:23:10 +00:00
/* This is a first order decision, so we'd better save the
* direction we chose */
current - > user_data [ NPF_TRACKDIR_CHOICE ] = trackdir ;
2005-04-11 19:53:44 +00:00
DEBUG ( npf , 6 ) ( " Saving trackdir: %#x " , trackdir ) ;
2005-01-31 11:23:10 +00:00
} else {
/* We've already made the decision, so just save our parent's
* decision */
current - > user_data [ NPF_TRACKDIR_CHOICE ] = parent - > path . node . user_data [ NPF_TRACKDIR_CHOICE ] ;
}
}
/* Will return the cost of the tunnel. If it is an entry, it will return the
* cost of that tile . If the tile is an exit , it will return the tunnel length
* including the exit tile . Requires that this is a Tunnel tile */
2005-06-25 19:08:58 +00:00
static uint NPFTunnelCost ( AyStarNode * current )
{
2005-06-17 17:11:05 +00:00
DiagDirection exitdir = TrackdirToExitdir ( ( Trackdir ) current - > direction ) ;
2005-01-31 11:23:10 +00:00
TileIndex tile = current - > tile ;
2005-10-05 07:20:26 +00:00
if ( ( DiagDirection ) GB ( _m [ tile ] . m5 , 0 , 2 ) = = ReverseDiagdir ( exitdir ) ) {
2005-01-31 11:23:10 +00:00
/* We just popped out if this tunnel, since were
* facing the tunnel exit */
FindLengthOfTunnelResult flotr ;
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
flotr = FindLengthOfTunnel ( tile , ReverseDiagdir ( exitdir ) ) ;
2005-01-31 11:23:10 +00:00
return flotr . length * NPF_TILE_LENGTH ;
//TODO: Penalty for tunnels?
} else {
/* We are entering the tunnel, the enter tile is just a
* straight track */
return NPF_TILE_LENGTH ;
}
}
2005-06-25 19:08:58 +00:00
static uint NPFSlopeCost ( AyStarNode * current )
{
2005-06-17 00:22:46 +00:00
TileIndex next = current - > tile + TileOffsByDir ( TrackdirToExitdir ( current - > direction ) ) ;
2005-03-14 16:56:05 +00:00
int x , y ;
int8 z1 , z2 ;
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
x = TileX ( current - > tile ) * TILE_SIZE ;
y = TileY ( current - > tile ) * TILE_SIZE ;
/* get the height of the center of the current tile */
z1 = GetSlopeZ ( x + TILE_HEIGHT , y + TILE_HEIGHT ) ;
2005-03-14 16:56:05 +00:00
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
x = TileX ( next ) * TILE_SIZE ;
y = TileY ( next ) * TILE_SIZE ;
/* get the height of the center of the next tile */
z2 = GetSlopeZ ( x + TILE_HEIGHT , y + TILE_HEIGHT ) ;
2005-03-14 16:56:05 +00:00
if ( ( z2 - z1 ) > 1 ) {
2005-01-31 11:23:10 +00:00
/* Slope up */
return _patches . npf_rail_slope_penalty ;
}
return 0 ;
/* Should we give a bonus for slope down? Probably not, we
* could just substract that bonus from the penalty , because
* there is only one level of steepness . . . */
}
2005-04-11 19:53:44 +00:00
/* Mark tiles by mowing the grass when npf debug level >= 1 */
2005-06-25 19:08:58 +00:00
static void NPFMarkTile ( TileIndex tile )
{
2005-04-11 19:53:44 +00:00
# ifdef NO_DEBUG_MESSAGES
return ;
# else
if ( _debug_npf_level > = 1 )
switch ( GetTileType ( tile ) ) {
case MP_RAILWAY :
/* DEBUG: mark visited tiles by mowing the grass under them
* ; - ) */
2005-05-03 00:11:30 +00:00
if ( ! IsTileDepotType ( tile , TRANSPORT_RAIL ) ) {
2005-10-05 07:20:26 +00:00
SB ( _m [ tile ] . m2 , 0 , 4 , 0 ) ;
2005-05-03 00:11:30 +00:00
MarkTileDirtyByTile ( tile ) ;
}
break ;
case MP_STREET :
if ( ! IsTileDepotType ( tile , TRANSPORT_ROAD ) ) {
2005-10-05 07:20:26 +00:00
SB ( _m [ tile ] . m2 , 4 , 3 , 0 ) ;
2005-05-03 00:11:30 +00:00
MarkTileDirtyByTile ( tile ) ;
}
2005-04-11 19:53:44 +00:00
break ;
default :
break ;
}
# endif
2005-01-31 11:23:10 +00:00
}
2005-06-25 19:08:58 +00:00
static int32 NPFWaterPathCost ( AyStar * as , AyStarNode * current , OpenListNode * parent )
{
2005-01-31 11:23:10 +00:00
//TileIndex tile = current->tile;
int32 cost = 0 ;
2005-06-17 17:11:05 +00:00
Trackdir trackdir = ( Trackdir ) current - > direction ;
2005-01-31 11:23:10 +00:00
cost = _trackdir_length [ trackdir ] ; /* Should be different for diagonal tracks */
2005-06-17 17:11:05 +00:00
if ( IsBuoyTile ( current - > tile ) & & IsDiagonalTrackdir ( trackdir ) )
2005-05-02 22:13:20 +00:00
cost + = _patches . npf_buoy_penalty ; /* A small penalty for going over buoys */
2005-06-17 17:11:05 +00:00
if ( current - > direction ! = NextTrackdir ( ( Trackdir ) parent - > path . node . direction ) )
2005-05-02 22:13:20 +00:00
cost + = _patches . npf_water_curve_penalty ;
/* TODO More penalties? */
2005-01-31 11:23:10 +00:00
return cost ;
}
/* Determine the cost of this node, for road tracks */
2005-06-25 19:08:58 +00:00
static int32 NPFRoadPathCost ( AyStar * as , AyStarNode * current , OpenListNode * parent )
{
2005-01-31 11:23:10 +00:00
TileIndex tile = current - > tile ;
int32 cost = 0 ;
2005-05-07 22:00:36 +00:00
2005-01-31 11:23:10 +00:00
/* Determine base length */
switch ( GetTileType ( tile ) ) {
case MP_TUNNELBRIDGE :
2005-10-05 07:20:26 +00:00
if ( GB ( _m [ tile ] . m5 , 4 , 4 ) = = 0 ) {
2005-01-31 11:23:10 +00:00
cost = NPFTunnelCost ( current ) ;
break ;
}
2005-11-05 18:32:21 +00:00
cost = NPF_TILE_LENGTH ;
break ;
2005-01-31 11:23:10 +00:00
case MP_STREET :
cost = NPF_TILE_LENGTH ;
2005-07-03 13:02:54 +00:00
/* Increase the cost for level crossings */
2005-11-05 18:32:21 +00:00
if ( IsLevelCrossing ( tile ) )
2005-07-03 13:02:54 +00:00
cost + = _patches . npf_crossing_penalty ;
2005-01-31 11:23:10 +00:00
break ;
default :
break ;
}
/* Determine extra costs */
/* Check for slope */
cost + = NPFSlopeCost ( current ) ;
2005-06-16 17:53:40 +00:00
/* Check for turns. Road vehicles only really drive diagonal, turns are
* represented by non - diagonal tracks */
if ( ! IsDiagonalTrackdir ( current - > direction ) )
cost + = _patches . npf_road_curve_penalty ;
2005-01-31 11:23:10 +00:00
NPFMarkTile ( tile ) ;
2005-04-11 19:53:44 +00:00
DEBUG ( npf , 4 ) ( " Calculating G for: (%d, %d). Result: %d " , TileX ( current - > tile ) , TileY ( current - > tile ) , cost ) ;
2005-01-31 11:23:10 +00:00
return cost ;
}
/* Determine the cost of this node, for railway tracks */
2005-06-25 19:08:58 +00:00
static int32 NPFRailPathCost ( AyStar * as , AyStarNode * current , OpenListNode * parent )
{
2005-01-31 11:23:10 +00:00
TileIndex tile = current - > tile ;
2005-06-17 17:11:05 +00:00
Trackdir trackdir = ( Trackdir ) current - > direction ;
2005-01-31 11:23:10 +00:00
int32 cost = 0 ;
2005-05-07 22:00:36 +00:00
/* HACK: We create a OpenListNode manualy, so we can call EndNodeCheck */
2005-04-02 10:38:31 +00:00
OpenListNode new_node ;
2005-01-31 11:23:10 +00:00
/* Determine base length */
switch ( GetTileType ( tile ) ) {
case MP_TUNNELBRIDGE :
2005-10-05 07:20:26 +00:00
if ( GB ( _m [ tile ] . m5 , 4 , 4 ) = = 0 ) {
2005-01-31 11:23:10 +00:00
cost = NPFTunnelCost ( current ) ;
break ;
}
/* Fall through if above if is false, it is a bridge
* then . We treat that as ordinary rail */
case MP_RAILWAY :
cost = _trackdir_length [ trackdir ] ; /* Should be different for diagonal tracks */
break ;
case MP_STREET : /* Railway crossing */
cost = NPF_TILE_LENGTH ;
break ;
2005-03-14 16:56:05 +00:00
case MP_STATION :
/* We give a station tile a penalty. Logically we would only
* want to give station tiles that are not our destination
* this penalty . This would discourage trains to drive through
* busy stations . But , we can just give any station tile a
* penalty , because every possible route will get this penalty
* exactly once , on its end tile ( if it ' s a station ) and it
* will therefore not make a difference . */
cost = NPF_TILE_LENGTH + _patches . npf_rail_station_penalty ;
break ;
2005-01-31 11:23:10 +00:00
default :
break ;
}
/* Determine extra costs */
2005-07-04 14:58:55 +00:00
/* Check for reserved tracks (PBS) */
if ( ( as - > user_data [ NPF_PBS_MODE ] ! = PBS_MODE_NONE ) & & ! ( NPFGetFlag ( current , NPF_FLAG_PBS_EXIT ) ) & & ! ( NPFGetFlag ( current , NPF_FLAG_PBS_BLOCKED ) ) & & ( PBSTileUnavail ( tile ) & ( 1 < < trackdir ) ) ) {
NPFSetFlag ( current , NPF_FLAG_PBS_BLOCKED , true ) ;
} ;
2005-03-08 19:54:10 +00:00
/* Check for signals */
2005-06-17 00:22:46 +00:00
if ( IsTileType ( tile , MP_RAILWAY ) & & HasSignalOnTrackdir ( tile , trackdir ) ) {
2005-03-08 19:54:10 +00:00
/* Ordinary track with signals */
2005-06-17 00:22:46 +00:00
if ( GetSignalState ( tile , trackdir ) = = SIGNAL_STATE_RED ) {
2005-01-31 11:23:10 +00:00
/* Signal facing us is red */
2005-03-08 19:54:10 +00:00
if ( ! NPFGetFlag ( current , NPF_FLAG_SEEN_SIGNAL ) ) {
2005-01-31 11:23:10 +00:00
/* Penalize the first signal we
* encounter , if it is red */
2005-04-04 17:53:39 +00:00
/* Is this a presignal exit or combo? */
2005-06-17 00:46:33 +00:00
SignalType sigtype = GetSignalType ( tile , TrackdirToTrack ( trackdir ) ) ;
2005-06-17 00:22:46 +00:00
if ( sigtype = = SIGTYPE_EXIT | | sigtype = = SIGTYPE_COMBO )
2005-04-04 17:53:39 +00:00
/* Penalise exit and combo signals differently (heavier) */
cost + = _patches . npf_rail_firstred_exit_penalty ;
else
cost + = _patches . npf_rail_firstred_penalty ;
2005-07-04 14:58:55 +00:00
/* for pbs runs, store the fact that the exit signal to the pbs block was red */
if ( ! ( NPFGetFlag ( current , NPF_FLAG_PBS_EXIT ) ) & & ! ( NPFGetFlag ( current , NPF_FLAG_PBS_RED ) ) & & NPFGetFlag ( current , NPF_FLAG_PBS_CHOICE ) )
NPFSetFlag ( current , NPF_FLAG_PBS_RED , true ) ;
2005-01-31 11:23:10 +00:00
}
2005-03-08 19:54:10 +00:00
/* Record the state of this signal */
NPFSetFlag ( current , NPF_FLAG_LAST_SIGNAL_RED , true ) ;
} else {
/* Record the state of this signal */
NPFSetFlag ( current , NPF_FLAG_LAST_SIGNAL_RED , false ) ;
2005-01-31 11:23:10 +00:00
}
2005-07-04 14:58:55 +00:00
if ( ! NPFGetFlag ( current , NPF_FLAG_SEEN_SIGNAL ) & & NPFGetFlag ( current , NPF_FLAG_PBS_BLOCKED ) ) {
/* penalise a path through the pbs block if it crosses reserved tracks */
cost + = 1000 ;
}
if ( ( PBSIsPbsSignal ( tile , trackdir ) ) & & ! NPFGetFlag ( current , NPF_FLAG_SEEN_SIGNAL ) ) {
/* we've encountered an exit signal to the pbs block */
NPFSetFlag ( current , NPF_FLAG_PBS_EXIT , true ) ;
}
2005-03-08 19:54:10 +00:00
NPFSetFlag ( current , NPF_FLAG_SEEN_SIGNAL , true ) ;
2005-01-31 11:23:10 +00:00
}
2005-03-08 19:54:10 +00:00
/* Penalise the tile if it is a target tile and the last signal was
* red */
2005-06-17 17:11:05 +00:00
/* HACK: We create a new_node here so we can call EndNodeCheck. Ugly as hell
* of course . . . */
2005-04-02 10:38:31 +00:00
new_node . path . node = * current ;
2005-06-17 17:11:05 +00:00
if ( as - > EndNodeCheck ( as , & new_node ) = = AYSTAR_FOUND_END_NODE & & NPFGetFlag ( current , NPF_FLAG_LAST_SIGNAL_RED ) )
2005-03-08 19:54:10 +00:00
cost + = _patches . npf_rail_lastred_penalty ;
2005-01-31 11:23:10 +00:00
/* Check for slope */
cost + = NPFSlopeCost ( current ) ;
/* Check for turns */
2005-06-17 17:11:05 +00:00
if ( current - > direction ! = NextTrackdir ( ( Trackdir ) parent - > path . node . direction ) )
2005-03-08 19:59:56 +00:00
cost + = _patches . npf_rail_curve_penalty ;
//TODO, with realistic acceleration, also the amount of straight track between
// curves should be taken into account, as this affects the speed limit.
2005-01-31 11:23:10 +00:00
2005-07-04 14:58:55 +00:00
/* Check for depots */
if ( IsTileDepotType ( tile , TRANSPORT_RAIL ) ) {
2005-05-07 22:00:36 +00:00
/* Penalise any depot tile that is not the last tile in the path. This
* _should_ penalise every occurence of reversing in a depot ( and only
* that ) */
2005-07-04 14:58:55 +00:00
if ( as - > EndNodeCheck ( as , & new_node ) ! = AYSTAR_FOUND_END_NODE )
cost + = _patches . npf_rail_depot_reverse_penalty ;
/* Do we treat this depot as a pbs signal? */
if ( ! NPFGetFlag ( current , NPF_FLAG_SEEN_SIGNAL ) ) {
if ( NPFGetFlag ( current , NPF_FLAG_PBS_BLOCKED ) ) {
cost + = 1000 ;
}
2005-07-22 08:40:19 +00:00
if ( PBSIsPbsSegment ( tile , ReverseTrackdir ( trackdir ) ) ) {
2005-07-04 14:58:55 +00:00
NPFSetFlag ( current , NPF_FLAG_PBS_EXIT , true ) ;
NPFSetFlag ( current , NPF_FLAG_SEEN_SIGNAL , true ) ;
}
}
NPFSetFlag ( current , NPF_FLAG_LAST_SIGNAL_RED , false ) ;
}
2005-05-07 22:00:36 +00:00
2005-01-31 11:23:10 +00:00
/* Check for occupied track */
//TODO
NPFMarkTile ( tile ) ;
2005-04-11 19:53:44 +00:00
DEBUG ( npf , 4 ) ( " Calculating G for: (%d, %d). Result: %d " , TileX ( current - > tile ) , TileY ( current - > tile ) , cost ) ;
2005-01-31 11:23:10 +00:00
return cost ;
}
/* Will find any depot */
2005-06-25 19:08:58 +00:00
static int32 NPFFindDepot ( AyStar * as , OpenListNode * current )
{
2005-04-02 10:38:31 +00:00
TileIndex tile = current - > path . node . tile ;
2005-05-07 22:00:36 +00:00
/* It's not worth caching the result with NPF_FLAG_IS_TARGET here as below,
* since checking the cache not that much faster than the actual check */
2005-02-06 22:36:08 +00:00
if ( IsTileDepotType ( tile , as - > user_data [ NPF_TYPE ] ) )
2005-01-31 11:23:10 +00:00
return AYSTAR_FOUND_END_NODE ;
else
return AYSTAR_DONE ;
}
/* Will find a station identified using the NPFFindStationOrTileData */
2005-06-25 19:08:58 +00:00
static int32 NPFFindStationOrTile ( AyStar * as , OpenListNode * current )
{
2005-03-08 22:28:59 +00:00
NPFFindStationOrTileData * fstd = ( NPFFindStationOrTileData * ) as - > user_target ;
2005-04-02 10:38:31 +00:00
AyStarNode * node = & current - > path . node ;
2005-03-08 22:28:59 +00:00
TileIndex tile = node - > tile ;
2005-03-08 19:54:10 +00:00
2005-01-31 11:23:10 +00:00
/* If GetNeighbours said we could get here, we assume the station type
* is correct */
2005-03-08 19:54:10 +00:00
if (
( fstd - > station_index = = - 1 & & tile = = fstd - > dest_coords ) | | /* We've found the tile, or */
2005-07-13 18:04:01 +00:00
( IsTileType ( tile , MP_STATION ) & & _m [ tile ] . m2 = = fstd - > station_index ) | | /* the station */
2005-07-04 14:58:55 +00:00
( NPFGetFlag ( node , NPF_FLAG_PBS_TARGET_SEEN ) ) /* or, we've passed it already (for pbs) */
2005-03-08 19:54:10 +00:00
) {
2005-07-04 14:58:55 +00:00
NPFSetFlag ( & current - > path . node , NPF_FLAG_PBS_TARGET_SEEN , true ) ;
/* for pbs runs, only accept we've found the target if we've also found a way out of the block */
2005-07-09 11:39:35 +00:00
if ( ( as - > user_data [ NPF_PBS_MODE ] ! = PBS_MODE_NONE ) & & ! NPFGetFlag ( node , NPF_FLAG_SEEN_SIGNAL ) & & ! IsEndOfLine ( node - > tile , node - > direction , as - > user_data [ NPF_RAILTYPE ] ) )
2005-07-04 14:58:55 +00:00
return AYSTAR_DONE ;
2005-03-08 19:54:10 +00:00
return AYSTAR_FOUND_END_NODE ;
} else {
2005-01-31 11:23:10 +00:00
return AYSTAR_DONE ;
2005-03-08 19:54:10 +00:00
}
2005-01-31 11:23:10 +00:00
}
/* To be called when current contains the (shortest route to) the target node.
* Will fill the contents of the NPFFoundTargetData using
* AyStarNode [ NPF_TRACKDIR_CHOICE ] .
*/
2005-06-25 19:08:58 +00:00
static void NPFSaveTargetData ( AyStar * as , OpenListNode * current )
{
2005-01-31 11:23:10 +00:00
NPFFoundTargetData * ftd = ( NPFFoundTargetData * ) as - > user_path ;
2005-06-17 17:11:05 +00:00
ftd - > best_trackdir = ( Trackdir ) current - > path . node . user_data [ NPF_TRACKDIR_CHOICE ] ;
2005-01-31 11:23:10 +00:00
ftd - > best_path_dist = current - > g ;
ftd - > best_bird_dist = 0 ;
ftd - > node = current - > path . node ;
2005-07-04 14:58:55 +00:00
ftd - > path = current - > path ;
2005-01-31 11:23:10 +00:00
}
2005-06-22 22:38:18 +00:00
/**
* Finds out if a given player ' s vehicles are allowed to enter a given tile .
* @ param owner The owner of the vehicle .
* @ param tile The tile that is about to be entered .
* @ param enterdir The direction from which the vehicle wants to enter the tile .
* @ return true if the vehicle can enter the tile .
* @ todo This function should be used in other places than just NPF ,
* maybe moved to another file too .
*/
2005-06-25 19:08:58 +00:00
static bool VehicleMayEnterTile ( Owner owner , TileIndex tile , DiagDirection enterdir )
2005-06-22 22:38:18 +00:00
{
if (
IsTileType ( tile , MP_RAILWAY ) /* Rail tile (also rail depot) */
| | IsTrainStationTile ( tile ) /* Rail station tile */
| | IsTileDepotType ( tile , TRANSPORT_ROAD ) /* Road depot tile */
| | IsRoadStationTile ( tile ) /* Road station tile */
| | IsTileDepotType ( tile , TRANSPORT_WATER ) /* Water depot tile */
)
return IsTileOwner ( tile , owner ) ; /* You need to own these tiles entirely to use them */
switch ( GetTileType ( tile ) ) {
case MP_STREET :
/* rail-road crossing : are we looking at the railway part? */
if ( IsLevelCrossing ( tile ) & & GetCrossingTransportType ( tile , TrackdirToTrack ( DiagdirToDiagTrackdir ( enterdir ) ) ) = = TRANSPORT_RAIL )
return IsTileOwner ( tile , owner ) ; /* Railway needs owner check, while the street is public */
break ;
case MP_TUNNELBRIDGE :
#if 0
/* OPTIMISATION: If we are on the middle of a bridge, we will not do the cpu
* intensive owner check , instead we will just assume that if the vehicle
* managed to get on the bridge , it is probably allowed to : - )
*/
2005-10-05 07:20:26 +00:00
if ( ( _m [ tile ] . m5 & 0xC6 ) = = 0xC0 & & GB ( _m [ tile ] . m5 , 0 , 1 ) = = ( enterdir & 0x1 ) ) {
2005-06-22 22:38:18 +00:00
/* on the middle part of a railway bridge: find bridge ending */
2005-07-13 18:04:01 +00:00
while ( IsTileType ( tile , MP_TUNNELBRIDGE ) & & ! ( ( _m [ tile ] . m5 & 0xC6 ) = = 0x80 ) ) {
2005-10-05 07:20:26 +00:00
tile + = TileOffsByDir ( GB ( _m [ tile ] . m5 , 0 , 1 ) ) ;
2005-06-22 22:38:18 +00:00
}
}
/* if we were on a railway middle part, we are now at a railway bridge ending */
# endif
if (
2005-07-13 18:04:01 +00:00
( _m [ tile ] . m5 & 0xFC ) = = 0 /* railway tunnel */
| | ( _m [ tile ] . m5 & 0xC6 ) = = 0x80 /* railway bridge ending */
2005-10-05 07:20:26 +00:00
| | ( ( _m [ tile ] . m5 & 0xF8 ) = = 0xE0 & & GB ( _m [ tile ] . m5 , 0 , 1 ) ! = ( enterdir & 0x1 ) ) /* railway under bridge */
2005-06-22 22:38:18 +00:00
)
return IsTileOwner ( tile , owner ) ;
break ;
default :
break ;
}
return true ; /* no need to check */
}
2005-01-31 11:23:10 +00:00
/* Will just follow the results of GetTileTrackStatus concerning where we can
* go and where not . Uses AyStar . user_data [ NPF_TYPE ] as the transport type and
* an argument to GetTileTrackStatus . Will skip tunnels , meaning that the
2005-03-08 19:54:10 +00:00
* entry and exit are neighbours . Will fill
* AyStarNode . user_data [ NPF_TRACKDIR_CHOICE ] with an appropriate value , and
* copy AyStarNode . user_data [ NPF_NODE_FLAGS ] from the parent */
2005-06-25 19:08:58 +00:00
static void NPFFollowTrack ( AyStar * aystar , OpenListNode * current )
{
2005-06-17 17:11:05 +00:00
Trackdir src_trackdir = ( Trackdir ) current - > path . node . direction ;
2005-01-31 11:23:10 +00:00
TileIndex src_tile = current - > path . node . tile ;
2005-06-17 00:22:46 +00:00
DiagDirection src_exitdir = TrackdirToExitdir ( src_trackdir ) ;
2005-01-31 11:23:10 +00:00
FindLengthOfTunnelResult flotr ;
TileIndex dst_tile ;
2005-06-17 17:11:05 +00:00
int i ;
2005-06-17 00:22:46 +00:00
TrackdirBits trackdirbits , ts ;
2005-01-31 11:23:10 +00:00
TransportType type = aystar - > user_data [ NPF_TYPE ] ;
/* Initialize to 0, so we can jump out (return) somewhere an have no neighbours */
aystar - > num_neighbours = 0 ;
2005-04-11 19:53:44 +00:00
DEBUG ( npf , 4 ) ( " Expanding: (%d, %d, %d) [%d] " , TileX ( src_tile ) , TileY ( src_tile ) , src_trackdir , src_tile ) ;
2005-01-31 11:23:10 +00:00
2005-07-04 14:58:55 +00:00
aystar - > EndNodeCheck ( aystar , current ) ;
2005-01-31 11:23:10 +00:00
/* Find dest tile */
2005-10-05 07:20:26 +00:00
if ( IsTileType ( src_tile , MP_TUNNELBRIDGE ) & & GB ( _m [ src_tile ] . m5 , 4 , 4 ) = = 0 & &
( DiagDirection ) GB ( _m [ src_tile ] . m5 , 0 , 2 ) = = src_exitdir ) {
2005-01-31 11:23:10 +00:00
/* This is a tunnel. We know this tunnel is our type,
* otherwise we wouldn ' t have got here . It is also facing us ,
* so we should skip it ' s body */
flotr = FindLengthOfTunnel ( src_tile , src_exitdir ) ;
dst_tile = flotr . tile ;
} else {
2005-04-05 22:06:02 +00:00
if ( type ! = TRANSPORT_WATER & & ( IsRoadStationTile ( src_tile ) | | IsTileDepotType ( src_tile , type ) ) ) {
2005-01-31 11:23:10 +00:00
/* This is a road station or a train or road depot. We can enter and exit
* those from one side only . Trackdirs don ' t support that ( yet ) , so we ' ll
* do this here . */
2005-06-17 17:11:05 +00:00
DiagDirection exitdir ;
2005-01-31 11:23:10 +00:00
/* Find out the exit direction first */
if ( IsRoadStationTile ( src_tile ) )
exitdir = GetRoadStationDir ( src_tile ) ;
else /* Train or road depot. Direction is stored the same for both, in map5 */
2005-04-05 22:06:02 +00:00
exitdir = GetDepotDirection ( src_tile , type ) ;
2005-01-31 11:23:10 +00:00
2005-05-07 22:00:36 +00:00
/* Let's see if were headed the right way into the depot, and reverse
* otherwise ( only for trains , since only with trains you can
* ( sometimes ) reach tiles after reversing that you couldn ' t reach
* without reversing . */
2005-06-17 00:22:46 +00:00
if ( src_trackdir = = DiagdirToDiagTrackdir ( ReverseDiagdir ( exitdir ) ) & & type = = TRANSPORT_RAIL )
2005-04-04 18:36:08 +00:00
/* We are headed inwards. We can only reverse here, so we'll not
* consider this direction , but jump ahead to the reverse direction .
* It would be nicer to return one neighbour here ( the reverse
* trackdir of the one we are considering now ) and then considering
* that one to return the tracks outside of the depot . But , because
* the code layout is cleaner this way , we will just pretend we are
* reversed already */
2005-06-17 00:22:46 +00:00
src_trackdir = ReverseTrackdir ( src_trackdir ) ;
2005-01-31 11:23:10 +00:00
}
/* This a normal tile, a bridge, a tunnel exit, etc. */
2005-06-17 00:22:46 +00:00
dst_tile = AddTileIndexDiffCWrap ( src_tile , TileIndexDiffCByDir ( TrackdirToExitdir ( src_trackdir ) ) ) ;
2005-01-31 11:23:10 +00:00
if ( dst_tile = = INVALID_TILE ) {
/* We reached the border of the map */
/* TODO Nicer control flow for this */
return ;
}
}
2005-06-22 01:06:04 +00:00
/* I can't enter a tunnel entry/exit tile from a tile above the tunnel. Note
* that I can enter the tunnel from a tile below the tunnel entrance . This
* solves the problem of vehicles wanting to drive off a tunnel entrance */
2005-10-05 07:20:26 +00:00
if ( IsTileType ( dst_tile , MP_TUNNELBRIDGE ) & & GB ( _m [ dst_tile ] . m5 , 4 , 4 ) = = 0 & &
GetTileZ ( dst_tile ) < GetTileZ ( src_tile ) ) {
2005-06-22 01:06:04 +00:00
return ;
2005-10-05 07:20:26 +00:00
}
2005-06-22 01:06:04 +00:00
2005-07-03 13:02:54 +00:00
/* check correct rail type (mono, maglev, etc) */
2005-05-02 18:53:06 +00:00
if ( type = = TRANSPORT_RAIL ) {
2005-07-03 13:02:54 +00:00
RailType dst_type = GetTileRailType ( dst_tile , src_trackdir ) ;
if ( ! IsCompatibleRail ( aystar - > user_data [ NPF_RAILTYPE ] , dst_type ) )
2005-05-02 18:53:06 +00:00
return ;
}
2005-02-06 22:36:08 +00:00
/* Check the owner of the tile */
2005-06-22 22:38:18 +00:00
if ( ! VehicleMayEnterTile ( aystar - > user_data [ NPF_OWNER ] , dst_tile , TrackdirToExitdir ( src_trackdir ) ) ) {
return ;
}
2005-01-31 11:23:10 +00:00
/* Determine available tracks */
2005-04-06 21:03:24 +00:00
if ( type ! = TRANSPORT_WATER & & ( IsRoadStationTile ( dst_tile ) | | IsTileDepotType ( dst_tile , type ) ) ) {
/* Road stations and road and train depots return 0 on GTTS, so we have to do this by hand... */
2005-06-17 17:11:05 +00:00
DiagDirection exitdir ;
2005-02-06 22:36:08 +00:00
if ( IsRoadStationTile ( dst_tile ) )
exitdir = GetRoadStationDir ( dst_tile ) ;
2005-04-06 21:03:24 +00:00
else /* Road or train depot */
2005-04-05 22:06:02 +00:00
exitdir = GetDepotDirection ( dst_tile , type ) ;
/* Find the trackdirs that are available for a depot or station with this
* orientation . They are only " inwards " , since we are reaching this tile
* from some other tile . This prevents vehicles driving into depots from
* the back */
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
ts = TrackdirToTrackdirBits ( DiagdirToDiagTrackdir ( ReverseDiagdir ( exitdir ) ) ) ;
2005-01-31 11:23:10 +00:00
} else {
ts = GetTileTrackStatus ( dst_tile , type ) ;
}
2005-07-04 14:58:55 +00:00
trackdirbits = ts & TRACKDIR_BIT_MASK ; /* Filter out signal status and the unused bits */
2005-01-31 11:23:10 +00:00
2005-06-17 00:22:46 +00:00
DEBUG ( npf , 4 ) ( " Next node: (%d, %d) [%d], possible trackdirs: %#x " , TileX ( dst_tile ) , TileY ( dst_tile ) , dst_tile , trackdirbits ) ;
2005-01-31 11:23:10 +00:00
/* Select only trackdirs we can reach from our current trackdir */
2005-06-17 00:22:46 +00:00
trackdirbits & = TrackdirReachesTrackdirs ( src_trackdir ) ;
2005-01-31 11:23:10 +00:00
if ( _patches . forbid_90_deg & & ( type = = TRANSPORT_RAIL | | type = = TRANSPORT_WATER ) ) /* Filter out trackdirs that would make 90 deg turns for trains */
2005-07-04 14:58:55 +00:00
trackdirbits & = ~ TrackdirCrossesTrackdirs ( src_trackdir ) ;
if ( KillFirstBit2x64 ( trackdirbits ) ! = 0 )
NPFSetFlag ( & current - > path . node , NPF_FLAG_PBS_CHOICE , true ) ;
/* When looking for 'any' route, ie when already inside a pbs block, discard all tracks that would cross
other reserved tracks , so we * always * will find a valid route if there is one */
if ( ! ( NPFGetFlag ( & current - > path . node , NPF_FLAG_PBS_EXIT ) ) & & ( aystar - > user_data [ NPF_PBS_MODE ] = = PBS_MODE_ANY ) )
trackdirbits & = ~ PBSTileUnavail ( dst_tile ) ;
2005-06-17 00:22:46 +00:00
DEBUG ( npf , 6 ) ( " After filtering: (%d, %d), possible trackdirs: %#x " , TileX ( dst_tile ) , TileY ( dst_tile ) , trackdirbits ) ;
2005-01-31 11:23:10 +00:00
2005-06-17 17:11:05 +00:00
i = 0 ;
2005-01-31 11:23:10 +00:00
/* Enumerate possible track */
2005-06-17 00:22:46 +00:00
while ( trackdirbits ! = 0 ) {
2005-06-17 17:11:05 +00:00
Trackdir dst_trackdir ;
2005-06-17 00:22:46 +00:00
dst_trackdir = FindFirstBit2x64 ( trackdirbits ) ;
trackdirbits = KillFirstBit2x64 ( trackdirbits ) ;
DEBUG ( npf , 5 ) ( " Expanded into trackdir: %d, remaining trackdirs: %#x " , dst_trackdir , trackdirbits ) ;
2005-01-31 11:23:10 +00:00
/* Check for oneway signal against us */
2005-06-17 00:22:46 +00:00
if ( IsTileType ( dst_tile , MP_RAILWAY ) & & GetRailTileType ( dst_tile ) = = RAIL_TYPE_SIGNALS ) {
if ( HasSignalOnTrackdir ( dst_tile , ReverseTrackdir ( dst_trackdir ) ) & & ! HasSignalOnTrackdir ( dst_tile , dst_trackdir ) )
2005-01-31 11:23:10 +00:00
// if one way signal not pointing towards us, stop going in this direction.
2005-06-17 00:22:46 +00:00
break ;
2005-01-31 11:23:10 +00:00
}
{
/* We've found ourselves a neighbour :-) */
AyStarNode * neighbour = & aystar - > neighbours [ i ] ;
neighbour - > tile = dst_tile ;
neighbour - > direction = dst_trackdir ;
/* Save user data */
neighbour - > user_data [ NPF_NODE_FLAGS ] = current - > path . node . user_data [ NPF_NODE_FLAGS ] ;
NPFFillTrackdirChoice ( neighbour , current ) ;
}
i + + ;
}
aystar - > num_neighbours = i ;
}
/*
* Plan a route to the specified target ( which is checked by target_proc ) ,
* from start1 and if not NULL , from start2 as well . The type of transport we
2005-05-07 22:00:36 +00:00
* are checking is in type . reverse_penalty is applied to all routes that
* originate from the second start node .
2005-01-31 11:23:10 +00:00
* When we are looking for one specific target ( optionally multiple tiles ) , we
* should use a good heuristic to perform aystar search . When we search for
* multiple targets that are spread around , we should perform a breadth first
* search by specifiying CalcZero as our heuristic .
*/
2005-07-04 14:58:55 +00:00
static NPFFoundTargetData NPFRouteInternal ( AyStarNode * start1 , AyStarNode * start2 , NPFFindStationOrTileData * target , AyStar_EndNodeCheck target_proc , AyStar_CalculateH heuristic_proc , TransportType type , Owner owner , RailType railtype , uint reverse_penalty , byte pbs_mode )
2005-06-25 19:08:58 +00:00
{
2005-01-31 11:23:10 +00:00
int r ;
NPFFoundTargetData result ;
/* Initialize procs */
_npf_aystar . CalculateH = heuristic_proc ;
_npf_aystar . EndNodeCheck = target_proc ;
_npf_aystar . FoundEndNode = NPFSaveTargetData ;
_npf_aystar . GetNeighbours = NPFFollowTrack ;
if ( type = = TRANSPORT_RAIL )
_npf_aystar . CalculateG = NPFRailPathCost ;
else if ( type = = TRANSPORT_ROAD )
_npf_aystar . CalculateG = NPFRoadPathCost ;
else if ( type = = TRANSPORT_WATER )
_npf_aystar . CalculateG = NPFWaterPathCost ;
else
assert ( 0 ) ;
2005-07-04 14:58:55 +00:00
if ( pbs_mode ! = PBS_MODE_NONE )
_npf_aystar . BeforeExit = NPFReservePBSPath ;
else
_npf_aystar . BeforeExit = NULL ;
2005-01-31 11:23:10 +00:00
/* Initialize Start Node(s) */
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start1 - > user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
start1 - > user_data [ NPF_NODE_FLAGS ] = 0 ;
2005-05-07 22:00:36 +00:00
_npf_aystar . addstart ( & _npf_aystar , start1 , 0 ) ;
2005-01-31 11:23:10 +00:00
if ( start2 ) {
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start2 - > user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-03-08 19:54:10 +00:00
start2 - > user_data [ NPF_NODE_FLAGS ] = 0 ;
NPFSetFlag ( start2 , NPF_FLAG_REVERSE , true ) ;
2005-05-07 22:00:36 +00:00
_npf_aystar . addstart ( & _npf_aystar , start2 , reverse_penalty ) ;
2005-01-31 11:23:10 +00:00
}
/* Initialize result */
result . best_bird_dist = ( uint ) - 1 ;
result . best_path_dist = ( uint ) - 1 ;
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
result . best_trackdir = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
_npf_aystar . user_path = & result ;
/* Initialize target */
_npf_aystar . user_target = target ;
/* Initialize user_data */
_npf_aystar . user_data [ NPF_TYPE ] = type ;
2005-02-06 22:36:08 +00:00
_npf_aystar . user_data [ NPF_OWNER ] = owner ;
2005-07-03 13:02:54 +00:00
_npf_aystar . user_data [ NPF_RAILTYPE ] = railtype ;
2005-07-04 14:58:55 +00:00
_npf_aystar . user_data [ NPF_PBS_MODE ] = pbs_mode ;
2005-01-31 11:23:10 +00:00
/* GO! */
r = AyStarMain_Main ( & _npf_aystar ) ;
assert ( r ! = AYSTAR_STILL_BUSY ) ;
if ( result . best_bird_dist ! = 0 ) {
if ( target ) {
DEBUG ( misc , 1 ) ( " NPF: Could not find route to 0x%x from 0x%x. " , target - > dest_coords , start1 - > tile ) ;
} else {
/* Assumption: target == NULL, so we are looking for a depot */
DEBUG ( misc , 1 ) ( " NPF: Could not find route to a depot from 0x%x. " , start1 - > tile ) ;
}
}
return result ;
}
2005-07-04 14:58:55 +00:00
NPFFoundTargetData NPFRouteToStationOrTileTwoWay ( TileIndex tile1 , Trackdir trackdir1 , TileIndex tile2 , Trackdir trackdir2 , NPFFindStationOrTileData * target , TransportType type , Owner owner , RailType railtype , byte pbs_mode )
2005-06-25 19:08:58 +00:00
{
2005-01-31 11:23:10 +00:00
AyStarNode start1 ;
AyStarNode start2 ;
start1 . tile = tile1 ;
start2 . tile = tile2 ;
2005-05-07 22:00:36 +00:00
/* We set this in case the target is also the start tile, we will just
* return a not found then */
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start1 . user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
start1 . direction = trackdir1 ;
start2 . direction = trackdir2 ;
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start2 . user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
2005-07-04 14:58:55 +00:00
return NPFRouteInternal ( & start1 , ( IsValidTile ( tile2 ) ? & start2 : NULL ) , target , NPFFindStationOrTile , NPFCalcStationOrTileHeuristic , type , owner , railtype , 0 , pbs_mode ) ;
2005-01-31 11:23:10 +00:00
}
2005-07-04 14:58:55 +00:00
NPFFoundTargetData NPFRouteToStationOrTile ( TileIndex tile , Trackdir trackdir , NPFFindStationOrTileData * target , TransportType type , Owner owner , RailType railtype , byte pbs_mode )
2005-06-25 19:08:58 +00:00
{
2005-07-04 14:58:55 +00:00
return NPFRouteToStationOrTileTwoWay ( tile , trackdir , INVALID_TILE , 0 , target , type , owner , railtype , pbs_mode ) ;
2005-01-31 11:23:10 +00:00
}
2005-07-03 13:02:54 +00:00
NPFFoundTargetData NPFRouteToDepotBreadthFirstTwoWay ( TileIndex tile1 , Trackdir trackdir1 , TileIndex tile2 , Trackdir trackdir2 , TransportType type , Owner owner , RailType railtype , uint reverse_penalty )
2005-06-25 19:08:58 +00:00
{
2005-05-07 22:00:36 +00:00
AyStarNode start1 ;
AyStarNode start2 ;
2005-01-31 11:23:10 +00:00
2005-05-07 22:00:36 +00:00
start1 . tile = tile1 ;
start2 . tile = tile2 ;
2005-01-31 11:23:10 +00:00
/* We set this in case the target is also the start tile, we will just
* return a not found then */
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start1 . user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-05-07 22:00:36 +00:00
start1 . direction = trackdir1 ;
start2 . direction = trackdir2 ;
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start2 . user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
/* perform a breadth first search. Target is NULL,
* since we are just looking for any depot . . . */
2005-07-04 14:58:55 +00:00
return NPFRouteInternal ( & start1 , ( IsValidTile ( tile2 ) ? & start2 : NULL ) , NULL , NPFFindDepot , NPFCalcZero , type , owner , railtype , reverse_penalty , PBS_MODE_NONE ) ;
2005-05-07 22:00:36 +00:00
}
2005-07-03 13:02:54 +00:00
NPFFoundTargetData NPFRouteToDepotBreadthFirst ( TileIndex tile , Trackdir trackdir , TransportType type , Owner owner , RailType railtype )
2005-06-25 19:08:58 +00:00
{
2005-07-03 13:02:54 +00:00
return NPFRouteToDepotBreadthFirstTwoWay ( tile , trackdir , INVALID_TILE , 0 , type , owner , railtype , 0 ) ;
2005-01-31 11:23:10 +00:00
}
2005-07-03 13:02:54 +00:00
NPFFoundTargetData NPFRouteToDepotTrialError ( TileIndex tile , Trackdir trackdir , TransportType type , Owner owner , RailType railtype )
2005-06-25 19:08:58 +00:00
{
2005-01-31 11:23:10 +00:00
/* Okay, what we're gonna do. First, we look at all depots, calculate
* the manhatten distance to get to each depot . We then sort them by
* distance . We start by trying to plan a route to the closest , then
* the next closest , etc . We stop when the best route we have found so
* far , is shorter than the manhattan distance . This will obviously
* always find the closest depot . It will probably be most efficient
* for ships , since the heuristic will not be to far off then . I hope .
*/
Queue depots ;
int r ;
NPFFoundTargetData best_result ;
NPFFoundTargetData result ;
NPFFindStationOrTileData target ;
AyStarNode start ;
Depot * current ;
2005-02-06 10:18:47 +00:00
Depot * depot ;
2005-01-31 11:23:10 +00:00
init_InsSort ( & depots ) ;
/* Okay, let's find all depots that we can use first */
2005-02-06 10:18:47 +00:00
FOR_ALL_DEPOTS ( depot ) {
2005-02-06 22:36:08 +00:00
/* Check if this is really a valid depot, it is of the needed type and
* owner */
2005-02-07 18:51:46 +00:00
if ( IsValidDepot ( depot ) & & IsTileDepotType ( depot - > xy , type ) & & IsTileOwner ( depot - > xy , owner ) )
2005-02-06 22:36:08 +00:00
/* If so, let's add it to the queue, sorted by distance */
2005-02-06 10:18:47 +00:00
depots . push ( & depots , depot , DistanceManhattan ( tile , depot - > xy ) ) ;
2005-01-31 11:23:10 +00:00
}
/* Now, let's initialise the aystar */
/* Initialize procs */
_npf_aystar . CalculateH = NPFCalcStationOrTileHeuristic ;
_npf_aystar . EndNodeCheck = NPFFindStationOrTile ;
_npf_aystar . FoundEndNode = NPFSaveTargetData ;
_npf_aystar . GetNeighbours = NPFFollowTrack ;
if ( type = = TRANSPORT_RAIL )
_npf_aystar . CalculateG = NPFRailPathCost ;
else if ( type = = TRANSPORT_ROAD )
_npf_aystar . CalculateG = NPFRoadPathCost ;
else if ( type = = TRANSPORT_WATER )
_npf_aystar . CalculateG = NPFWaterPathCost ;
else
assert ( 0 ) ;
2005-07-04 14:58:55 +00:00
_npf_aystar . BeforeExit = NULL ;
2005-01-31 11:23:10 +00:00
/* Initialize target */
target . station_index = - 1 ; /* We will initialize dest_coords inside the loop below */
_npf_aystar . user_target = & target ;
/* Initialize user_data */
_npf_aystar . user_data [ NPF_TYPE ] = type ;
2005-02-06 22:36:08 +00:00
_npf_aystar . user_data [ NPF_OWNER ] = owner ;
2005-07-04 14:58:55 +00:00
_npf_aystar . user_data [ NPF_PBS_MODE ] = PBS_MODE_NONE ;
2005-01-31 11:23:10 +00:00
/* Initialize Start Node */
start . tile = tile ;
start . direction = trackdir ; /* We will initialize user_data inside the loop below */
/* Initialize Result */
_npf_aystar . user_path = & result ;
best_result . best_path_dist = ( uint ) - 1 ;
2005-02-06 22:36:08 +00:00
best_result . best_bird_dist = ( uint ) - 1 ;
2005-01-31 11:23:10 +00:00
/* Just iterate the depots in order of increasing distance */
while ( ( current = depots . pop ( & depots ) ) ) {
/* Check to see if we already have a path shorter than this
2005-02-06 22:36:08 +00:00
* depot ' s manhattan distance . HACK : We call DistanceManhattan
2005-01-31 11:23:10 +00:00
* again , we should probably modify the queue to give us that
* value . . . */
if ( DistanceManhattan ( tile , current - > xy * NPF_TILE_LENGTH ) > best_result . best_path_dist )
break ;
/* Initialize Start Node */
/* We set this in case the target is also the start tile, we will just
* return a not found then */
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
start . user_data [ NPF_TRACKDIR_CHOICE ] = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
start . user_data [ NPF_NODE_FLAGS ] = 0 ;
2005-05-07 22:00:36 +00:00
_npf_aystar . addstart ( & _npf_aystar , & start , 0 ) ;
2005-01-31 11:23:10 +00:00
/* Initialize result */
result . best_bird_dist = ( uint ) - 1 ;
result . best_path_dist = ( uint ) - 1 ;
(svn r2448) General cleanup of rail related code, more to follow.
* Add: rail.[ch] for rail-related enums and wrapper functions.
* Codechange: Removed dozens of magic numbers with below enums.
* Codechange: Rewrote CheckTrackCombination().
* Add: TILE_SIZE, TILE_PIXELS and TILE_HEIGHT constants.
* Add: enums RailTileType, RailTileSubtype, SignalType to mask against the map arrays.
* Add: enums Track, TrackBits, Trackdir, TrackdirBits for railway track data. (Note that the old RAIL_BIT constants are replaced by TRACK_BIT ones).
* Add: enums Direction and DiagDirection
* Codechange: Moved a bunch of track(dir) related lookup arrays from npf.[ch] to rail.[ch].
* Codechange: move RailType enum from tile.h to rail.h.
* Add: Wrapper functions for masking signal status in the map arrays: SignalAlongTrackdir, SignalAgainstTrackdir and SignalOnTrack.
* Add: Wrapper functions to access rail tiles, using above enums
* Add: Wrapper functions to modify tracks, trackdirs, directions, etc.
* Add: Wrapper functions for all lookup arrays in rail.[ch] (Arrays are still used in parts of the code)
* Codechange: Renamed some variables and arguments to better represent what they contain (railbit -> track, bits -> trackdirbits, etc.).
* Codechange: Don't use FindLandscapeHeight() in CmdRemoveSingleRail(), since it returns way too much info. Use GetTileSlope() instead.
* Codechange: [NPF] Removed some unused globals and code from npf.c.
2005-06-16 18:04:02 +00:00
result . best_trackdir = INVALID_TRACKDIR ;
2005-01-31 11:23:10 +00:00
/* Initialize target */
target . dest_coords = current - > xy ;
/* GO! */
r = AyStarMain_Main ( & _npf_aystar ) ;
assert ( r ! = AYSTAR_STILL_BUSY ) ;
/* This depot is closer */
if ( result . best_path_dist < best_result . best_path_dist )
best_result = result ;
}
if ( result . best_bird_dist ! = 0 ) {
DEBUG ( misc , 1 ) ( " NPF: Could not find route to any depot from 0x%x. " , tile ) ;
}
return best_result ;
}
void InitializeNPF ( void )
{
2005-04-07 19:19:16 +00:00
init_AyStar ( & _npf_aystar , NPFHash , NPF_HASH_SIZE ) ;
2005-03-08 22:21:20 +00:00
_npf_aystar . loops_per_tick = 0 ;
_npf_aystar . max_path_cost = 0 ;
2005-04-15 15:28:01 +00:00
//_npf_aystar.max_search_nodes = 0;
/* We will limit the number of nodes for now, until we have a better
* solution to really fix performance */
_npf_aystar . max_search_nodes = _patches . npf_max_search_nodes ;
2005-01-31 11:23:10 +00:00
}
2005-06-25 19:08:58 +00:00
void NPFFillWithOrderData ( NPFFindStationOrTileData * fstd , Vehicle * v )
{
2005-01-31 11:23:10 +00:00
/* Ships don't really reach their stations, but the tile in front. So don't
* save the station id for ships . For roadvehs we don ' t store it either ,
* because multistop depends on vehicles actually reaching the exact
* dest_tile , not just any stop of that station .
* So only for train orders to stations we fill fstd - > station_index , for all
* others only dest_coords */
if ( ( v - > current_order . type ) = = OT_GOTO_STATION & & v - > type = = VEH_Train ) {
fstd - > station_index = v - > current_order . station ;
2005-03-07 23:28:27 +00:00
/* Let's take the closest tile of the station as our target for trains */
fstd - > dest_coords = CalcClosestStationTile ( v - > current_order . station , v - > tile ) ;
2005-01-31 11:23:10 +00:00
} else {
fstd - > dest_coords = v - > dest_tile ;
fstd - > station_index = - 1 ;
}
}