OpenTTD Source 20260218-master-g2123fca5ea
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 <https://www.gnu.org/licenses/old-licenses/gpl-2.0>.
6 */
7
9
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
23enum class RailTileType : uint8_t {
24 Normal = 0,
25 Signals = 1,
26 Depot = 3,
27};
28
36[[debug_inline]] inline static RailTileType GetRailTileType(Tile t)
37{
38 assert(IsTileType(t, TileType::Railway));
39 return static_cast<RailTileType>(GB(t.m5(), 6, 2));
40}
41
49[[debug_inline]] inline static bool IsPlainRail(Tile t)
50{
52 return rtt == RailTileType::Normal || rtt == RailTileType::Signals;
53}
54
60[[debug_inline]] inline static bool IsPlainRailTile(Tile t)
61{
63}
64
65
72inline bool HasSignals(Tile t)
73{
75}
76
83inline void SetHasSignals(Tile tile, bool signals)
84{
85 assert(IsPlainRailTile(tile));
87}
88
95[[debug_inline]] inline static bool IsRailDepot(Tile t)
96{
98}
99
105[[debug_inline]] inline static bool IsRailDepotTile(Tile t)
106{
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
293{
294 return s == SIGTYPE_PBS || s == SIGTYPE_PBS_ONEWAY;
295}
296
305{
307 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 4 : 0;
308 return (SignalType)GB(t.m2(), pos, 3);
309}
310
318inline void SetSignalType(Tile t, Track track, SignalType s)
319{
321 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 4 : 0;
322 SB(t.m2(), pos, 3, s);
323 if (track == INVALID_TRACK) SB(t.m2(), 4, 3, s);
324}
325
333inline bool IsPresignalEntry(Tile t, Track track)
334{
335 return GetSignalType(t, track) == SIGTYPE_ENTRY || GetSignalType(t, track) == SIGTYPE_COMBO;
336}
337
345inline bool IsPresignalExit(Tile t, Track track)
346{
347 return GetSignalType(t, track) == SIGTYPE_EXIT || GetSignalType(t, track) == SIGTYPE_COMBO;
348}
349
358inline bool IsOnewaySignal(Tile t, Track track)
359{
360 return GetSignalType(t, track) != SIGTYPE_PBS;
361}
362
369inline void CycleSignalSide(Tile t, Track track)
370{
371 uint8_t sig;
372 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 4 : 6;
373
374 sig = GB(t.m3(), pos, 2);
375 if (--sig == 0) sig = IsPbsSignal(GetSignalType(t, track)) ? 2 : 3;
376 SB(t.m3(), pos, 2, sig);
377}
378
387{
388 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 7 : 3;
389 return (SignalVariant)GB(t.m2(), pos, 1);
390}
391
399inline void SetSignalVariant(Tile t, Track track, SignalVariant v)
400{
401 uint8_t pos = (track == TRACK_LOWER || track == TRACK_RIGHT) ? 7 : 3;
402 SB(t.m2(), pos, 1, v);
403 if (track == INVALID_TRACK) SB(t.m2(), 7, 1, v);
404}
405
411inline void SetSignalStates(Tile tile, uint state)
412{
413 SB(tile.m4(), 4, 4, state);
414}
415
421inline uint GetSignalStates(Tile tile)
422{
423 return GB(tile.m4(), 4, 4);
424}
425
432inline SignalState GetSingleSignalState(Tile t, uint8_t signalbit)
433{
434 return (SignalState)HasBit(GetSignalStates(t), signalbit);
435}
436
442inline void SetPresentSignals(Tile tile, uint signals)
443{
444 SB(tile.m3(), 4, 4, signals);
445}
446
452inline uint GetPresentSignals(Tile tile)
453{
454 return GB(tile.m3(), 4, 4);
455}
456
463inline bool IsSignalPresent(Tile t, uint8_t signalbit)
464{
465 return HasBit(GetPresentSignals(t), signalbit);
466}
467
475inline bool HasSignalOnTrack(Tile tile, Track track)
476{
477 assert(IsValidTrack(track));
478 return GetRailTileType(tile) == RailTileType::Signals && (GetPresentSignals(tile) & SignalOnTrack(track)) != 0;
479}
480
491inline bool HasSignalOnTrackdir(Tile tile, Trackdir trackdir)
492{
493 assert (IsValidTrackdir(trackdir));
495}
496
507{
508 assert(IsValidTrackdir(trackdir));
509 assert(HasSignalOnTrack(tile, TrackdirToTrack(trackdir)));
510 return GetSignalStates(tile) & SignalAlongTrackdir(trackdir) ?
512}
513
520inline void SetSignalStateByTrackdir(Tile tile, Trackdir trackdir, SignalState state)
521{
522 if (state == SIGNAL_STATE_GREEN) { // set 1
523 SetSignalStates(tile, GetSignalStates(tile) | SignalAlongTrackdir(trackdir));
524 } else {
525 SetSignalStates(tile, GetSignalStates(tile) & ~SignalAlongTrackdir(trackdir));
526 }
527}
528
536{
537 return IsTileType(tile, TileType::Railway) && HasSignalOnTrackdir(tile, td) &&
539}
540
549{
551 !HasSignalOnTrackdir(tile, td) && IsOnewaySignal(tile, TrackdirToTrack(td));
552}
553
554
556
575
582{
583 SB(t.m4(), 0, 4, to_underlying(rgt));
584}
585
592{
593 return static_cast<RailGroundType>(GB(t.m4(), 0, 4));
594}
595
605
606
614inline void MakeRailNormal(Tile t, Owner o, TrackBits b, RailType r)
615{
617 SetTileOwner(t, o);
618 SetDockingTile(t, false);
619 t.m2() = 0;
620 t.m3() = 0;
621 t.m4() = 0;
622 t.m5() = to_underlying(RailTileType::Normal) << 6 | b;
623 SB(t.m6(), 2, 6, 0);
624 t.m7() = 0;
625 t.m8() = r;
626}
627
634{
635 assert(IsRailDepotTile(tile));
636 SB(tile.m5(), 0, 2, dir);
637}
638
647inline void MakeRailDepot(Tile tile, Owner owner, DepotID depot_id, DiagDirection dir, RailType rail_type)
648{
650 SetTileOwner(tile, owner);
651 SetDockingTile(tile, false);
652 tile.m2() = depot_id.base();
653 tile.m3() = 0;
654 tile.m4() = 0;
655 tile.m5() = to_underlying(RailTileType::Depot) << 6 | dir;
656 SB(tile.m6(), 2, 6, 0);
657 tile.m7() = 0;
658 tile.m8() = rail_type;
659}
660
661#endif /* RAIL_MAP_H */
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.
static constexpr uint GB(const T x, const uint8_t s, const uint8_t n)
Fetch n bits from x, started at bit s.
constexpr bool HasBit(const T x, const uint8_t y)
Checks if a bit in a value is set.
Wrapper class to abstract away the way the tiles are stored.
Definition map_func.h:25
uint8_t & m5()
General purpose.
Definition map_func.h:156
uint8_t & m4()
General purpose.
Definition map_func.h:145
uint8_t & m6()
General purpose.
Definition map_func.h:167
uint8_t & m7()
Primarily used for newgrf support.
Definition map_func.h:178
uint8_t & m3()
General purpose.
Definition map_func.h:134
uint16_t & m8()
General purpose.
Definition map_func.h:189
uint16_t & m2()
Primarily used for indices to towns, industries and stations.
Definition map_func.h:123
Header files for depots (not hangars).
PoolID< uint16_t, struct DepotIDTag, 64000, 0xFFFF > DepotID
Type for the unique identifier of depots.
Definition depot_type.h:15
DiagDirection
Enumeration for diagonal directions.
constexpr std::underlying_type_t< enum_type > to_underlying(enum_type e)
Implementation of std::to_underlying (from C++23).
Definition enum_type.hpp:21
bool HasOnewaySignalBlockingTrackdir(Tile tile, Trackdir td)
Is a one-way signal blocking the trackdir?
Definition rail_map.h:548
static RailTileType GetRailTileType(Tile t)
Returns the RailTileType (normal with or without signals, waypoint or depot).
Definition rail_map.h:36
static bool IsPlainRail(Tile t)
Returns whether this is plain rails, with or without signals.
Definition rail_map.h:49
uint GetSignalStates(Tile tile)
Set the states of the signals (Along/AgainstTrackDir).
Definition rail_map.h:421
void MakeRailDepot(Tile tile, Owner owner, DepotID depot_id, DiagDirection dir, RailType rail_type)
Make a rail depot.
Definition rail_map.h:647
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:491
static bool IsRailDepot(Tile t)
Is this rail tile a rail depot?
Definition rail_map.h:95
SignalVariant GetSignalVariant(Tile t, Track track)
Get the signal variant for a track on a tile.
Definition rail_map.h:386
RailGroundType GetRailGroundType(Tile t)
Get the ground type for rail tiles.
Definition rail_map.h:591
TrackBits GetTrackBits(Tile tile)
Gets the track bits of the given tile.
Definition rail_map.h:136
static bool IsPlainRailTile(Tile t)
Checks whether the tile is a rail tile or rail tile with signals.
Definition rail_map.h:60
bool IsPbsSignal(SignalType s)
Checks whether the given signal is a path based signal.
Definition rail_map.h:292
void CycleSignalSide(Tile t, Track track)
Cycle to the next signal side at the given track on a tile.
Definition rail_map.h:369
void MakeRailNormal(Tile t, Owner o, TrackBits b, RailType r)
Make the given tile a normal rail.
Definition rail_map.h:614
RailTileType
Different types of Rail-related tiles.
Definition rail_map.h:23
@ Normal
Normal rail tile without signals.
Definition rail_map.h:24
@ Depot
Depot (one entrance).
Definition rail_map.h:26
@ Signals
Normal rail tile with signals.
Definition rail_map.h:25
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:463
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:558
@ FenceSENW
Grass with a fence at the NW and SE edges.
Definition rail_map.h:563
@ FenceVert2
Grass with a fence at the western side.
Definition rail_map.h:568
@ HalfTileSnow
Snow only on higher part of slope (steep or one corner raised).
Definition rail_map.h:573
@ Barren
Nothing (dirt).
Definition rail_map.h:559
@ FenceNESW
Grass with a fence at the NE and SW edges.
Definition rail_map.h:566
@ FenceHoriz2
Grass with a fence at the northern side.
Definition rail_map.h:570
@ SnowOrDesert
Icy or sandy.
Definition rail_map.h:571
@ HalfTileWater
Grass with a fence and shore or water on the free halftile.
Definition rail_map.h:572
@ FenceVert1
Grass with a fence at the eastern side.
Definition rail_map.h:567
@ FenceNW
Grass with a fence at the NW edge.
Definition rail_map.h:561
@ FenceSE
Grass with a fence at the SE edge.
Definition rail_map.h:562
@ FenceNE
Grass with a fence at the NE edge.
Definition rail_map.h:564
@ Grass
Grassy.
Definition rail_map.h:560
@ FenceSW
Grass with a fence at the SW edge.
Definition rail_map.h:565
@ FenceHoriz1
Grass with a fence at the southern side.
Definition rail_map.h:569
void SetSignalStateByTrackdir(Tile tile, Trackdir trackdir, SignalState state)
Sets the state of the signal along the given trackdir.
Definition rail_map.h:520
uint GetPresentSignals(Tile tile)
Get whether the given signals are present (Along/AgainstTrackDir).
Definition rail_map.h:452
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:475
bool HasPbsSignalOnTrackdir(Tile tile, Trackdir td)
Is a pbs signal present along the trackdir?
Definition rail_map.h:535
bool IsOnewaySignal(Tile t, Track track)
Is the signal at the given track on a tile a one way signal?
Definition rail_map.h:358
void SetPresentSignals(Tile tile, uint signals)
Set whether the given signals are present (Along/AgainstTrackDir).
Definition rail_map.h:442
bool IsPresignalExit(Tile t, Track track)
Is the signal at the given track on a tile a presignal exit signal?
Definition rail_map.h:345
void SetRailGroundType(Tile t, RailGroundType rgt)
Set the ground type for rail tiles.
Definition rail_map.h:581
void SetRailDepotExitDirection(Tile tile, DiagDirection dir)
Sets the exit direction of a rail depot.
Definition rail_map.h:633
SignalType GetSignalType(Tile t, Track track)
Get the signal type for a track on a tile.
Definition rail_map.h:304
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:37
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
TrackBits GetRailReservationTrackBits(Tile t)
Returns the reserved track bits of the tile.
Definition rail_map.h:194
void SetSignalType(Tile t, Track track, SignalType s)
Set the signal type for a track on a tile.
Definition rail_map.h:318
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 IsSnowOrDesertRailGround(Tile t)
Is the given rail tile snowy or deserty.
Definition rail_map.h:601
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
SignalState GetSignalStateByTrackdir(Tile tile, Trackdir trackdir)
Gets the state of the signal along the given trackdir.
Definition rail_map.h:506
void SetSignalVariant(Tile t, Track track, SignalVariant v)
Set the signal variant for a track on a tile.
Definition rail_map.h:399
static bool IsRailDepotTile(Tile t)
Is this tile rail tile and a rail depot?
Definition rail_map.h:105
bool IsPresignalEntry(Tile t, Track track)
Is the signal at the given track on a tile a presignal entry signal?
Definition rail_map.h:333
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:432
void SetSignalStates(Tile tile, uint state)
Set the states of the signals (Along/AgainstTrackDir).
Definition rail_map.h:411
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.
static bool IsTileType(Tile tile, TileType type)
Checks if a tile is a given tiletype.
Definition tile_map.h:150
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
@ Railway
A tile with railway.
Definition tile_type.h:50
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:364