OpenTTD Source 20250528-master-g3aca5d62a8
rail_map.h
Go to the documentation of this file.
1/*
2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
6 */
7
10#ifndef RAIL_MAP_H
11#define RAIL_MAP_H
12
13#include "rail_type.h"
14#include "depot_type.h"
15#include "signal_func.h"
16#include "track_func.h"
17#include "tile_map.h"
18#include "water_map.h"
19#include "signal_type.h"
20
21
28
36debug_inline static RailTileType GetRailTileType(Tile t)
37{
38 assert(IsTileType(t, MP_RAILWAY));
39 return (RailTileType)GB(t.m5(), 6, 2);
40}
41
49debug_inline static bool IsPlainRail(Tile t)
50{
52 return rtt == RAIL_TILE_NORMAL || rtt == RAIL_TILE_SIGNALS;
53}
54
60debug_inline static bool IsPlainRailTile(Tile t)
61{
62 return IsTileType(t, MP_RAILWAY) && IsPlainRail(t);
63}
64
65
72inline bool HasSignals(Tile t)
73{
75}
76
83inline void SetHasSignals(Tile tile, bool signals)
84{
85 assert(IsPlainRailTile(tile));
86 AssignBit(tile.m5(), 6, signals);
87}
88
95debug_inline static bool IsRailDepot(Tile t)
96{
98}
99
105debug_inline static bool IsRailDepotTile(Tile t)
106{
107 return IsTileType(t, MP_RAILWAY) && IsRailDepot(t);
108}
109
116{
117 return (RailType)GB(t.m8(), 0, 6);
118}
119
125inline void SetRailType(Tile t, RailType r)
126{
127 SB(t.m8(), 0, 6, r);
128}
129
130
137{
138 assert(IsPlainRailTile(tile));
139 return (TrackBits)GB(tile.m5(), 0, 6);
140}
141
147inline void SetTrackBits(Tile t, TrackBits b)
148{
149 assert(IsPlainRailTile(t));
150 SB(t.m5(), 0, 6, b);
151}
152
160inline bool HasTrack(Tile tile, Track track)
161{
162 return HasBit(GetTrackBits(tile), track);
163}
164
172{
173 return (DiagDirection)GB(t.m5(), 0, 2);
174}
175
186
187
195{
196 assert(IsPlainRailTile(t));
197 uint8_t track_b = GB(t.m2(), 8, 3);
198 Track track = (Track)(track_b - 1); // map array saves Track+1
199 if (track_b == 0) return TRACK_BIT_NONE;
200 return (TrackBits)(TrackToTrackBits(track) | (HasBit(t.m2(), 11) ? TrackToTrackBits(TrackToOppositeTrack(track)) : 0));
201}
202
210{
211 assert(IsPlainRailTile(t));
212 assert(!TracksOverlap(b));
213 Track track = RemoveFirstTrack(&b);
214 SB(t.m2(), 8, 3, track == INVALID_TRACK ? 0 : track + 1);
215 AssignBit(t.m2(), 11, b != TRACK_BIT_NONE);
216}
217
225inline bool TryReserveTrack(Tile tile, Track t)
226{
227 assert(HasTrack(tile, t));
228 TrackBits bits = TrackToTrackBits(t);
230 if ((res & bits) != TRACK_BIT_NONE) return false; // already reserved
231 res |= bits;
232 if (TracksOverlap(res)) return false; // crossing reservation present
233 SetTrackReservation(tile, res);
234 return true;
235}
236
243inline void UnreserveTrack(Tile tile, Track t)
244{
245 assert(HasTrack(tile, t));
247 res &= ~TrackToTrackBits(t);
248 SetTrackReservation(tile, res);
249}
250
258{
259 assert(IsRailDepot(t));
260 return HasBit(t.m5(), 4);
261}
262
269inline void SetDepotReservation(Tile t, bool b)
270{
271 assert(IsRailDepot(t));
272 AssignBit(t.m5(), 4, b);
273}
274
285
286
287inline bool IsPbsSignal(SignalType s)
288{
289 return s == SIGTYPE_PBS || s == SIGTYPE_PBS_ONEWAY;
290}
291
292inline SignalType GetSignalType(Tile t, Track track)
293{
295 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 4 : 0;
296 return (SignalType)GB(t.m2(), pos, 3);
297}
298
299inline void SetSignalType(Tile t, Track track, SignalType s)
300{
302 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 4 : 0;
303 SB(t.m2(), pos, 3, s);
304 if (track == INVALID_TRACK) SB(t.m2(), 4, 3, s);
305}
306
307inline bool IsPresignalEntry(Tile t, Track track)
308{
309 return GetSignalType(t, track) == SIGTYPE_ENTRY || GetSignalType(t, track) == SIGTYPE_COMBO;
310}
311
312inline bool IsPresignalExit(Tile t, Track track)
313{
314 return GetSignalType(t, track) == SIGTYPE_EXIT || GetSignalType(t, track) == SIGTYPE_COMBO;
315}
316
318inline bool IsOnewaySignal(Tile t, Track track)
319{
320 return GetSignalType(t, track) != SIGTYPE_PBS;
321}
322
323inline void CycleSignalSide(Tile t, Track track)
324{
325 uint8_t sig;
326 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 4 : 6;
327
328 sig = GB(t.m3(), pos, 2);
329 if (--sig == 0) sig = IsPbsSignal(GetSignalType(t, track)) ? 2 : 3;
330 SB(t.m3(), pos, 2, sig);
331}
332
333inline SignalVariant GetSignalVariant(Tile t, Track track)
334{
335 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 7 : 3;
336 return (SignalVariant)GB(t.m2(), pos, 1);
337}
338
339inline void SetSignalVariant(Tile t, Track track, SignalVariant v)
340{
341 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 7 : 3;
342 SB(t.m2(), pos, 1, v);
343 if (track == INVALID_TRACK) SB(t.m2(), 7, 1, v);
344}
345
351inline void SetSignalStates(Tile tile, uint state)
352{
353 SB(tile.m4(), 4, 4, state);
354}
355
361inline uint GetSignalStates(Tile tile)
362{
363 return GB(tile.m4(), 4, 4);
364}
365
372inline SignalState GetSingleSignalState(Tile t, uint8_t signalbit)
373{
374 return (SignalState)HasBit(GetSignalStates(t), signalbit);
375}
376
382inline void SetPresentSignals(Tile tile, uint signals)
383{
384 SB(tile.m3(), 4, 4, signals);
385}
386
392inline uint GetPresentSignals(Tile tile)
393{
394 return GB(tile.m3(), 4, 4);
395}
396
403inline bool IsSignalPresent(Tile t, uint8_t signalbit)
404{
405 return HasBit(GetPresentSignals(t), signalbit);
406}
407
412inline bool HasSignalOnTrack(Tile tile, Track track)
413{
414 assert(IsValidTrack(track));
415 return GetRailTileType(tile) == RAIL_TILE_SIGNALS && (GetPresentSignals(tile) & SignalOnTrack(track)) != 0;
416}
417
425inline bool HasSignalOnTrackdir(Tile tile, Trackdir trackdir)
426{
427 assert (IsValidTrackdir(trackdir));
428 return GetRailTileType(tile) == RAIL_TILE_SIGNALS && GetPresentSignals(tile) & SignalAlongTrackdir(trackdir);
429}
430
438{
439 assert(IsValidTrackdir(trackdir));
440 assert(HasSignalOnTrack(tile, TrackdirToTrack(trackdir)));
441 return GetSignalStates(tile) & SignalAlongTrackdir(trackdir) ?
443}
444
448inline void SetSignalStateByTrackdir(Tile tile, Trackdir trackdir, SignalState state)
449{
450 if (state == SIGNAL_STATE_GREEN) { // set 1
451 SetSignalStates(tile, GetSignalStates(tile) | SignalAlongTrackdir(trackdir));
452 } else {
453 SetSignalStates(tile, GetSignalStates(tile) & ~SignalAlongTrackdir(trackdir));
454 }
455}
456
463{
464 return IsTileType(tile, MP_RAILWAY) && HasSignalOnTrackdir(tile, td) &&
465 IsPbsSignal(GetSignalType(tile, TrackdirToTrack(td)));
466}
467
475{
476 return IsTileType(tile, MP_RAILWAY) && HasSignalOnTrackdir(tile, ReverseTrackdir(td)) &&
477 !HasSignalOnTrackdir(tile, td) && IsOnewaySignal(tile, TrackdirToTrack(td));
478}
479
480
482
501
502inline void SetRailGroundType(Tile t, RailGroundType rgt)
503{
504 SB(t.m4(), 0, 4, rgt);
505}
506
507inline RailGroundType GetRailGroundType(Tile t)
508{
509 return (RailGroundType)GB(t.m4(), 0, 4);
510}
511
512inline bool IsSnowRailGround(Tile t)
513{
514 return GetRailGroundType(t) == RAIL_GROUND_ICE_DESERT;
515}
516
517
518inline void MakeRailNormal(Tile t, Owner o, TrackBits b, RailType r)
519{
521 SetTileOwner(t, o);
522 SetDockingTile(t, false);
523 t.m2() = 0;
524 t.m3() = 0;
525 t.m4() = 0;
526 t.m5() = RAIL_TILE_NORMAL << 6 | b;
527 SB(t.m6(), 2, 4, 0);
528 t.m7() = 0;
529 t.m8() = r;
530}
531
538{
539 assert(IsRailDepotTile(tile));
540 SB(tile.m5(), 0, 2, dir);
541}
542
551inline void MakeRailDepot(Tile tile, Owner owner, DepotID depot_id, DiagDirection dir, RailType rail_type)
552{
553 SetTileType(tile, MP_RAILWAY);
554 SetTileOwner(tile, owner);
555 SetDockingTile(tile, false);
556 tile.m2() = depot_id.base();
557 tile.m3() = 0;
558 tile.m4() = 0;
559 tile.m5() = RAIL_TILE_DEPOT << 6 | dir;
560 SB(tile.m6(), 2, 4, 0);
561 tile.m7() = 0;
562 tile.m8() = rail_type;
563}
564
565#endif /* RAIL_MAP_H */
debug_inline constexpr bool HasBit(const T x, const uint8_t y)
Checks if a bit in a value is set.
constexpr T AssignBit(T &x, const uint8_t y, bool value)
Assigns a bit in a variable.
constexpr T SB(T &x, const uint8_t s, const uint8_t n, const U d)
Set n bits in x starting at bit s to d.
debug_inline static constexpr uint GB(const T x, const uint8_t s, const uint8_t n)
Fetch n bits from x, started at bit s.
Wrapper class to abstract away the way the tiles are stored.
Definition map_func.h:25
debug_inline uint16_t & m8()
General purpose.
Definition map_func.h:197
debug_inline uint16_t & m2()
Primarily used for indices to towns, industries and stations.
Definition map_func.h:125
debug_inline uint8_t & m7()
Primarily used for newgrf support.
Definition map_func.h:185
debug_inline uint8_t & m4()
General purpose.
Definition map_func.h:149
debug_inline uint8_t & m6()
General purpose.
Definition map_func.h:173
debug_inline uint8_t & m3()
General purpose.
Definition map_func.h:137
debug_inline uint8_t & m5()
General purpose.
Definition map_func.h:161
Header files for depots (not hangars)
DiagDirection
Enumeration for diagonal directions.
bool HasOnewaySignalBlockingTrackdir(Tile tile, Trackdir td)
Is a one-way signal blocking the trackdir? A one-way signal on the trackdir against will block,...
Definition rail_map.h:474
uint GetSignalStates(Tile tile)
Set the states of the signals (Along/AgainstTrackDir)
Definition rail_map.h:361
void MakeRailDepot(Tile tile, Owner owner, DepotID depot_id, DiagDirection dir, RailType rail_type)
Make a rail depot.
Definition rail_map.h:551
RailType GetRailType(Tile t)
Gets the rail type of the given tile.
Definition rail_map.h:115
bool HasSignalOnTrackdir(Tile tile, Trackdir trackdir)
Checks for the presence of signals along the given trackdir on the given rail tile.
Definition rail_map.h:425
static debug_inline RailTileType GetRailTileType(Tile t)
Returns the RailTileType (normal with or without signals, waypoint or depot).
Definition rail_map.h:36
TrackBits GetTrackBits(Tile tile)
Gets the track bits of the given tile.
Definition rail_map.h:136
RailTileType
Different types of Rail-related tiles.
Definition rail_map.h:23
@ RAIL_TILE_DEPOT
Depot (one entrance)
Definition rail_map.h:26
@ RAIL_TILE_NORMAL
Normal rail tile without signals.
Definition rail_map.h:24
@ RAIL_TILE_SIGNALS
Normal rail tile with signals.
Definition rail_map.h:25
static debug_inline bool IsRailDepotTile(Tile t)
Is this tile rail tile and a rail depot?
Definition rail_map.h:105
Track GetRailDepotTrack(Tile t)
Returns the track of a depot, ignoring direction.
Definition rail_map.h:182
void SetTrackReservation(Tile t, TrackBits b)
Sets the reserved track bits of the tile.
Definition rail_map.h:209
bool IsSignalPresent(Tile t, uint8_t signalbit)
Checks whether the given signals is present.
Definition rail_map.h:403
DiagDirection GetRailDepotDirection(Tile t)
Returns the direction the depot is facing to.
Definition rail_map.h:171
void SetTrackBits(Tile t, TrackBits b)
Sets the track bits of the given tile.
Definition rail_map.h:147
RailGroundType
The ground 'under' the rail.
Definition rail_map.h:484
@ RAIL_GROUND_FENCE_HORIZ1
Grass with a fence at the southern side.
Definition rail_map.h:495
@ RAIL_GROUND_FENCE_VERT1
Grass with a fence at the eastern side.
Definition rail_map.h:493
@ RAIL_GROUND_ICE_DESERT
Icy or sandy.
Definition rail_map.h:497
@ RAIL_GROUND_FENCE_NE
Grass with a fence at the NE edge.
Definition rail_map.h:490
@ RAIL_GROUND_FENCE_NESW
Grass with a fence at the NE and SW edges.
Definition rail_map.h:492
@ RAIL_GROUND_FENCE_HORIZ2
Grass with a fence at the northern side.
Definition rail_map.h:496
@ RAIL_GROUND_FENCE_SW
Grass with a fence at the SW edge.
Definition rail_map.h:491
@ RAIL_GROUND_FENCE_NW
Grass with a fence at the NW edge.
Definition rail_map.h:487
@ RAIL_GROUND_WATER
Grass with a fence and shore or water on the free halftile.
Definition rail_map.h:498
@ RAIL_GROUND_BARREN
Nothing (dirt)
Definition rail_map.h:485
@ RAIL_GROUND_FENCE_VERT2
Grass with a fence at the western side.
Definition rail_map.h:494
@ RAIL_GROUND_GRASS
Grassy.
Definition rail_map.h:486
@ RAIL_GROUND_HALF_SNOW
Snow only on higher part of slope (steep or one corner raised)
Definition rail_map.h:499
@ RAIL_GROUND_FENCE_SENW
Grass with a fence at the NW and SE edges.
Definition rail_map.h:489
@ RAIL_GROUND_FENCE_SE
Grass with a fence at the SE edge.
Definition rail_map.h:488
void SetSignalStateByTrackdir(Tile tile, Trackdir trackdir, SignalState state)
Sets the state of the signal along the given trackdir.
Definition rail_map.h:448
uint GetPresentSignals(Tile tile)
Get whether the given signals are present (Along/AgainstTrackDir)
Definition rail_map.h:392
bool HasSignalOnTrack(Tile tile, Track track)
Checks for the presence of signals (either way) on the given track on the given rail tile.
Definition rail_map.h:412
bool HasPbsSignalOnTrackdir(Tile tile, Trackdir td)
Is a pbs signal present along the trackdir?
Definition rail_map.h:462
bool IsOnewaySignal(Tile t, Track track)
One-way signals can't be passed the 'wrong' way.
Definition rail_map.h:318
void SetPresentSignals(Tile tile, uint signals)
Set whether the given signals are present (Along/AgainstTrackDir)
Definition rail_map.h:382
static debug_inline bool IsRailDepot(Tile t)
Is this rail tile a rail depot?
Definition rail_map.h:95
void SetRailDepotExitDirection(Tile tile, DiagDirection dir)
Sets the exit direction of a rail depot.
Definition rail_map.h:537
void UnreserveTrack(Tile tile, Track t)
Lift the reservation of a specific track on a tile.
Definition rail_map.h:243
RailType GetTileRailType(Tile tile)
Return the rail type of tile, or INVALID_RAILTYPE if this is no rail tile.
Definition rail.cpp:155
void SetDepotReservation(Tile t, bool b)
Set the reservation state of the depot.
Definition rail_map.h:269
TrackBits GetDepotReservationTrackBits(Tile t)
Get the reserved track bits for a depot.
Definition rail_map.h:281
static debug_inline bool IsPlainRail(Tile t)
Returns whether this is plain rails, with or without signals.
Definition rail_map.h:49
TrackBits GetRailReservationTrackBits(Tile t)
Returns the reserved track bits of the tile.
Definition rail_map.h:194
bool HasDepotReservation(Tile t)
Get the reservation state of the depot.
Definition rail_map.h:257
void SetHasSignals(Tile tile, bool signals)
Add/remove the 'has signal' bit from the RailTileType.
Definition rail_map.h:83
bool TryReserveTrack(Tile tile, Track t)
Try to reserve a specific track on a tile.
Definition rail_map.h:225
bool HasSignals(Tile t)
Checks if a rail tile has signals.
Definition rail_map.h:72
static debug_inline bool IsPlainRailTile(Tile t)
Checks whether the tile is a rail tile or rail tile with signals.
Definition rail_map.h:60
SignalState GetSignalStateByTrackdir(Tile tile, Trackdir trackdir)
Gets the state of the signal along the given trackdir.
Definition rail_map.h:437
bool HasTrack(Tile tile, Track track)
Returns whether the given track is present on the given tile.
Definition rail_map.h:160
SignalState GetSingleSignalState(Tile t, uint8_t signalbit)
Get the state of a single signal.
Definition rail_map.h:372
void SetSignalStates(Tile tile, uint state)
Set the states of the signals (Along/AgainstTrackDir)
Definition rail_map.h:351
void SetRailType(Tile t, RailType r)
Sets the rail type of the given tile.
Definition rail_map.h:125
The different types of rail.
RailType
Enumeration for all possible railtypes.
Definition rail_type.h:25
Functions related to signals.
uint8_t SignalAlongTrackdir(Trackdir trackdir)
Maps a trackdir to the bit that stores its status in the map arrays, in the direction along with the ...
Definition signal_func.h:22
uint8_t SignalOnTrack(Track track)
Maps a Track to the bits that store the status of the two signals that can be present on the given tr...
Definition signal_func.h:42
Types and classes related to signals.
SignalType
Type of signal, i.e.
Definition signal_type.h:23
@ SIGTYPE_PBS_ONEWAY
no-entry signal
Definition signal_type.h:29
@ SIGTYPE_PBS
normal pbs signal
Definition signal_type.h:28
@ SIGTYPE_ENTRY
presignal block entry
Definition signal_type.h:25
@ SIGTYPE_COMBO
presignal inter-block
Definition signal_type.h:27
@ SIGTYPE_EXIT
presignal block exit
Definition signal_type.h:26
SignalState
These are states in which a signal can be.
Definition signal_type.h:42
@ SIGNAL_STATE_RED
The signal is red.
Definition signal_type.h:43
@ SIGNAL_STATE_GREEN
The signal is green.
Definition signal_type.h:44
SignalVariant
Variant of the signal, i.e.
Definition signal_type.h:16
Map writing/reading functions for tiles.
void SetTileType(Tile tile, TileType type)
Set the type of a tile.
Definition tile_map.h:131
void SetTileOwner(Tile tile, Owner owner)
Sets the owner of a tile.
Definition tile_map.h:198
static debug_inline bool IsTileType(Tile tile, TileType type)
Checks if a tile is a given tiletype.
Definition tile_map.h:150
@ MP_RAILWAY
A railway.
Definition tile_type.h:49
Different conversion functions from one kind of track to another.
Track TrackdirToTrack(Trackdir trackdir)
Returns the Track that a given Trackdir represents.
Definition track_func.h:262
Track TrackToOppositeTrack(Track t)
Find the opposite track to a given track.
Definition track_func.h:231
TrackBits TrackToTrackBits(Track track)
Maps a Track to the corresponding TrackBits value.
Definition track_func.h:77
Trackdir ReverseTrackdir(Trackdir trackdir)
Maps a trackdir to the reverse trackdir.
Definition track_func.h:247
bool TracksOverlap(TrackBits bits)
Checks if the given tracks overlap, ie form a crossing.
Definition track_func.h:645
bool IsValidTrackdir(Trackdir trackdir)
Checks if a Trackdir is valid for non-road vehicles.
Definition track_func.h:52
bool IsValidTrack(Track track)
Checks if a Track is valid.
Definition track_func.h:28
Track RemoveFirstTrack(TrackBits *tracks)
Removes first Track from TrackBits and returns it.
Definition track_func.h:131
Track DiagDirToDiagTrack(DiagDirection diagdir)
Maps a (4-way) direction to the diagonal track incidating with that diagdir.
Definition track_func.h:512
TrackBits
Allow incrementing of Track variables.
Definition track_type.h:35
@ TRACK_BIT_NONE
No track.
Definition track_type.h:36
Trackdir
Enumeration for tracks and directions.
Definition track_type.h:66
Track
These are used to specify a single track.
Definition track_type.h:19
@ INVALID_TRACK
Flag for an invalid track.
Definition track_type.h:28
@ TRACK_LOWER
Track in the lower corner of the tile (south)
Definition track_type.h:24
@ TRACK_RIGHT
Track in the right corner of the tile (east)
Definition track_type.h:26
Map accessors for water tiles.
void SetDockingTile(Tile t, bool b)
Set the docking tile state of a tile.
Definition water_map.h:361