OpenTTD Source 20250312-master-gcdcc6b491d
map_func.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 MAP_FUNC_H
11#define MAP_FUNC_H
12
13#include "core/math_func.hpp"
14#include "tile_type.h"
15#include "map_type.h"
16#include "direction_func.h"
17
25class Tile {
26private:
27 friend struct Map;
32 struct TileBase {
33 uint8_t type = 0;
34 uint8_t height = 0;
35 uint16_t m2 = 0;
36 uint8_t m1 = 0;
37 uint8_t m3 = 0;
38 uint8_t m4 = 0;
39 uint8_t m5 = 0;
40 };
41
42 static_assert(sizeof(TileBase) == 8);
43
48 struct TileExtended {
49 uint8_t m6 = 0;
50 uint8_t m7 = 0;
51 uint16_t m8 = 0;
52 };
53
54 static std::unique_ptr<TileBase[]> base_tiles;
55 static std::unique_ptr<TileExtended[]> extended_tiles;
56
58
59public:
64 debug_inline Tile(TileIndex tile) : tile(tile) {}
65
70 Tile(uint tile) : tile(tile) {}
71
75 debug_inline constexpr operator TileIndex() const { return this->tile; }
76
80 debug_inline constexpr operator uint() const { return this->tile.base(); }
81
89 debug_inline uint8_t &type()
90 {
91 return base_tiles[this->tile.base()].type;
92 }
93
101 debug_inline uint8_t &height()
102 {
103 return base_tiles[this->tile.base()].height;
104 }
105
113 debug_inline uint8_t &m1()
114 {
115 return base_tiles[this->tile.base()].m1;
116 }
117
125 debug_inline uint16_t &m2()
126 {
127 return base_tiles[this->tile.base()].m2;
128 }
129
137 debug_inline uint8_t &m3()
138 {
139 return base_tiles[this->tile.base()].m3;
140 }
141
149 debug_inline uint8_t &m4()
150 {
151 return base_tiles[this->tile.base()].m4;
152 }
153
161 debug_inline uint8_t &m5()
162 {
163 return base_tiles[this->tile.base()].m5;
164 }
165
173 debug_inline uint8_t &m6()
174 {
175 return extended_tiles[this->tile.base()].m6;
176 }
177
185 debug_inline uint8_t &m7()
186 {
187 return extended_tiles[this->tile.base()].m7;
188 }
189
197 debug_inline uint16_t &m8()
198 {
199 return extended_tiles[this->tile.base()].m8;
200 }
201};
202
206struct Map {
207private:
211 struct Iterator {
212 typedef Tile value_type;
213 typedef Tile *pointer;
214 typedef Tile &reference;
215 typedef size_t difference_type;
216 typedef std::forward_iterator_tag iterator_category;
217
218 explicit Iterator(TileIndex index) : index(index) {}
219 bool operator==(const Iterator &other) const { return this->index == other.index; }
220 Tile operator*() const { return this->index; }
221 Iterator & operator++() { this->index++; return *this; }
222 private:
223 TileIndex index;
224 };
225
226 /*
227 * Iterable ensemble of all Tiles
228 */
230 Iterator begin() { return Iterator(TileIndex{}); }
231 Iterator end() { return Iterator(TileIndex{Map::Size()}); }
232 bool empty() { return false; }
233 };
234
235 static uint log_x;
236 static uint log_y;
237 static uint size_x;
238 static uint size_y;
239 static uint size;
240 static uint tile_mask;
241
242public:
243 static void Allocate(uint size_x, uint size_y);
244
250 debug_inline static uint LogX()
251 {
252 return Map::log_x;
253 }
254
260 static inline uint LogY()
261 {
262 return Map::log_y;
263 }
264
269 debug_inline static uint SizeX()
270 {
271 return Map::size_x;
272 }
273
278 static inline uint SizeY()
279 {
280 return Map::size_y;
281 }
282
287 debug_inline static uint Size()
288 {
289 return Map::size;
290 }
291
296 debug_inline static uint MaxX()
297 {
298 return Map::SizeX() - 1;
299 }
300
305 static inline uint MaxY()
306 {
307 return Map::SizeY() - 1;
308 }
309
310
316 static inline TileIndex WrapToMap(TileIndex tile)
317 {
318 return TileIndex{tile.base() & Map::tile_mask};
319 }
320
327 static inline uint ScaleBySize(uint n)
328 {
329 /* Subtract 12 from shift in order to prevent integer overflow
330 * for large values of n. It's safe since the min mapsize is 64x64. */
331 return CeilDiv(n << (Map::LogX() + Map::LogY() - 12), 1 << 4);
332 }
333
340 static inline uint ScaleBySize1D(uint n)
341 {
342 /* Normal circumference for the X+Y is 256+256 = 1<<9
343 * Note, not actually taking the full circumference into account,
344 * just half of it. */
345 return CeilDiv((n << Map::LogX()) + (n << Map::LogY()), 1 << 9);
346 }
347
353 static bool IsInitialized()
354 {
355 return Tile::base_tiles != nullptr;
356 }
357
362 static IterateWrapper Iterate() { return IterateWrapper(); }
363};
364
372debug_inline static TileIndex TileXY(uint x, uint y)
373{
374 return TileIndex{(y << Map::LogX()) + x};
375}
376
388inline TileIndexDiff TileDiffXY(int x, int y)
389{
390 /* Multiplication gives much better optimization on MSVC than shifting.
391 * 0 << shift isn't optimized to 0 properly.
392 * Typically x and y are constants, and then this doesn't result
393 * in any actual multiplication in the assembly code.. */
394 return (y * Map::SizeX()) + x;
395}
396
403debug_inline static TileIndex TileVirtXY(uint x, uint y)
404{
405 return TileIndex{(y >> 4 << Map::LogX()) + (x >> 4)};
406}
407
408
414debug_inline static uint TileX(TileIndex tile)
415{
416 return tile.base() & Map::MaxX();
417}
418
424debug_inline static uint TileY(TileIndex tile)
425{
426 return tile.base() >> Map::LogX();
427}
428
440{
441 return TileDiffXY(tidc.x, tidc.y);
442}
443
444/* Helper functions to provide explicit +=/-= operators for TileIndex and TileIndexDiff. */
445constexpr TileIndex &operator+=(TileIndex &tile, TileIndexDiff offset) { tile = tile + TileIndex(offset); return tile; }
446constexpr TileIndex &operator-=(TileIndex &tile, TileIndexDiff offset) { tile = tile - TileIndex(offset); return tile; }
447
455#ifndef _DEBUG
456 constexpr TileIndex TileAdd(TileIndex tile, TileIndexDiff offset) { return tile + offset; }
457#else
459#endif
460
469inline TileIndex TileAddXY(TileIndex tile, int x, int y)
470{
471 return TileAdd(tile, TileDiffXY(x, y));
472}
473
474TileIndex TileAddWrap(TileIndex tile, int addx, int addy);
475
489
497{
499
500 assert(IsValidDirection(dir));
501 return _tileoffs_by_dir[dir];
502}
503
515{
516 int x = TileX(tile) + diff.x;
517 int y = TileY(tile) + diff.y;
518 /* Negative value will become big positive value after cast */
519 if ((uint)x >= Map::SizeX() || (uint)y >= Map::SizeY()) return INVALID_TILE;
520 return TileXY(x, y);
521}
522
531{
532 TileIndexDiffC difference;
533
534 difference.x = TileX(tile_a) - TileX(tile_b);
535 difference.y = TileY(tile_a) - TileY(tile_b);
536
537 return difference;
538}
539
540/* Functions to calculate distances */
547
555{
556 extern const TileIndexDiffC _tileoffs_by_axis[];
557
558 assert(IsValidAxis(axis));
560}
561
576
584{
586
587 assert(IsValidDirection(dir));
589}
590
599{
600 return TileAdd(tile, TileOffsByDir(dir));
601}
602
611{
612 return TileAdd(tile, TileOffsByDiagDir(dir));
613}
614
623{
624 int dx = (int)TileX(tile_to) - (int)TileX(tile_from);
625 int dy = (int)TileY(tile_to) - (int)TileY(tile_from);
626 if (dx == 0) {
627 if (dy == 0) return INVALID_DIAGDIR;
628 return (dy < 0 ? DIAGDIR_NW : DIAGDIR_SE);
629 } else {
630 if (dy != 0) return INVALID_DIAGDIR;
631 return (dx < 0 ? DIAGDIR_NE : DIAGDIR_SW);
632 }
633}
634
642typedef bool TestTileOnSearchProc(TileIndex tile, void *user_data);
643
644bool CircularTileSearch(TileIndex *tile, uint size, TestTileOnSearchProc proc, void *user_data);
645bool CircularTileSearch(TileIndex *tile, uint radius, uint w, uint h, TestTileOnSearchProc proc, void *user_data);
646
652inline TileIndex RandomTileSeed(uint32_t r)
653{
654 return Map::WrapToMap(TileIndex{r});
655}
656
663#define RandomTile() RandomTileSeed(Random())
664
665uint GetClosestWaterDistance(TileIndex tile, bool water);
666
667#endif /* MAP_FUNC_H */
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
static std::unique_ptr< TileExtended[]> extended_tiles
Pointer to the extended tile-array.
Definition map_func.h:55
debug_inline uint8_t & m7()
Primarily used for newgrf support.
Definition map_func.h:185
debug_inline Tile(TileIndex tile)
Create the tile wrapper for the given tile.
Definition map_func.h:64
debug_inline uint8_t & m4()
General purpose.
Definition map_func.h:149
static std::unique_ptr< TileBase[]> base_tiles
Pointer to the tile-array.
Definition map_func.h:54
debug_inline uint8_t & height()
The height of the northern corner.
Definition map_func.h:101
debug_inline uint8_t & m6()
General purpose.
Definition map_func.h:173
debug_inline uint8_t & type()
The type (bits 4..7), bridges (2..3), rainforest/desert (0..1)
Definition map_func.h:89
Tile(uint tile)
Create the tile wrapper for the given tile.
Definition map_func.h:70
debug_inline uint8_t & m3()
General purpose.
Definition map_func.h:137
debug_inline uint8_t & m1()
Primarily used for ownership information.
Definition map_func.h:113
debug_inline uint8_t & m5()
General purpose.
Definition map_func.h:161
TileIndex tile
The tile to access the map data for.
Definition map_func.h:57
Different functions related to conversions between directions.
bool IsValidAxis(Axis d)
Checks if an integer value is a valid Axis.
bool IsValidDiagDirection(DiagDirection d)
Checks if an integer value is a valid DiagDirection.
bool IsValidDirection(Direction d)
Checks if an integer value is a valid Direction.
Direction
Defines the 8 directions on the map.
@ DIR_END
Used to iterate.
Axis
Allow incrementing of DiagDirDiff variables.
DiagDirection
Enumeration for diagonal directions.
@ DIAGDIR_NE
Northeast, upper right on your monitor.
@ DIAGDIR_NW
Northwest.
@ DIAGDIR_SE
Southeast.
@ DIAGDIR_END
Used for iterations.
@ INVALID_DIAGDIR
Flag for an invalid DiagDirection.
@ DIAGDIR_SW
Southwest.
const TileIndexDiffC _tileoffs_by_axis[]
'Lookup table' for tile offsets given an Axis
const TileIndexDiffC _tileoffs_by_dir[]
'Lookup table' for tile offsets given a Direction
const TileIndexDiffC _tileoffs_by_diagdir[]
'Lookup table' for tile offsets given a DiagDirection
bool CircularTileSearch(TileIndex *tile, uint size, TestTileOnSearchProc proc, void *user_data)
Function performing a search around a center tile and going outward, thus in circle.
Definition map.cpp:243
DiagDirection DiagdirBetweenTiles(TileIndex tile_from, TileIndex tile_to)
Determines the DiagDirection to get from one tile to another.
Definition map_func.h:622
TileIndex TileAddXY(TileIndex tile, int x, int y)
Adds a given offset to a tile.
Definition map_func.h:469
uint DistanceMaxPlusManhattan(TileIndex, TileIndex)
Max + Manhattan.
Definition map.cpp:190
TileIndex AddTileIndexDiffCWrap(TileIndex tile, TileIndexDiffC diff)
Add a TileIndexDiffC to a TileIndex and returns the new one.
Definition map_func.h:514
TileIndex TileAddByDir(TileIndex tile, Direction dir)
Adds a Direction to a tile.
Definition map_func.h:598
static debug_inline TileIndex TileXY(uint x, uint y)
Returns the TileIndex of a coordinate.
Definition map_func.h:372
TileIndexDiff ToTileIndexDiff(TileIndexDiffC tidc)
Return the offset between two tiles from a TileIndexDiffC struct.
Definition map_func.h:439
TileIndex TileAddByDiagDir(TileIndex tile, DiagDirection dir)
Adds a DiagDir to a tile.
Definition map_func.h:610
uint DistanceMax(TileIndex, TileIndex)
also known as L-Infinity-Norm
Definition map.cpp:174
TileIndex RandomTileSeed(uint32_t r)
Get a random tile out of a given seed.
Definition map_func.h:652
TileIndexDiff TileDiffXY(int x, int y)
Calculates an offset for the given coordinate(-offset).
Definition map_func.h:388
TileIndexDiff TileOffsByAxis(Axis axis)
Convert an Axis to a TileIndexDiff.
Definition map_func.h:554
uint DistanceFromEdgeDir(TileIndex, DiagDirection)
distance from the map edge in given direction
Definition map.cpp:219
uint DistanceFromEdge(TileIndex)
shortest distance from any edge of the map
Definition map.cpp:202
TileIndex TileAddWrap(TileIndex tile, int addx, int addy)
This function checks if we add addx/addy to tile, if we do wrap around the edges.
Definition map.cpp:93
uint DistanceSquare(TileIndex, TileIndex)
Euclidean- or L2-Norm squared.
Definition map.cpp:159
TileIndexDiffC TileIndexDiffCByDiagDir(DiagDirection dir)
Returns the TileIndexDiffC offset from a DiagDirection.
Definition map_func.h:482
uint GetClosestWaterDistance(TileIndex tile, bool water)
Finds the distance for the closest tile with water/land given a tile.
Definition map.cpp:325
uint DistanceManhattan(TileIndex, TileIndex)
also known as L1-Norm. Is the shortest distance one could go over diagonal tracks (or roads)
Definition map.cpp:142
constexpr TileIndex TileAdd(TileIndex tile, TileIndexDiff offset)
Adds a given offset to a tile.
Definition map_func.h:456
bool TestTileOnSearchProc(TileIndex tile, void *user_data)
A callback function type for searching tiles.
Definition map_func.h:642
TileIndexDiffC TileIndexDiffCByDir(Direction dir)
Returns the TileIndexDiffC offset from a Direction.
Definition map_func.h:496
TileIndexDiff TileOffsByDir(Direction dir)
Convert a Direction to a TileIndexDiff.
Definition map_func.h:583
static debug_inline uint TileY(TileIndex tile)
Get the Y component of a tile.
Definition map_func.h:424
static debug_inline uint TileX(TileIndex tile)
Get the X component of a tile.
Definition map_func.h:414
TileIndexDiff TileOffsByDiagDir(DiagDirection dir)
Convert a DiagDirection to a TileIndexDiff.
Definition map_func.h:569
TileIndexDiffC TileIndexToTileIndexDiffC(TileIndex tile_a, TileIndex tile_b)
Returns the diff between two tiles.
Definition map_func.h:530
static debug_inline TileIndex TileVirtXY(uint x, uint y)
Get a tile from the virtual XY-coordinate.
Definition map_func.h:403
Types related to maps.
int32_t TileIndexDiff
An offset value between two tiles.
Definition map_type.h:23
Integer math functions.
constexpr uint CeilDiv(uint a, uint b)
Computes ceil(a / b) for non-negative a and b.
Iterator to iterate all Tiles.
Definition map_func.h:211
Size related data of the map.
Definition map_func.h:206
static TileIndex WrapToMap(TileIndex tile)
'Wraps' the given "tile" so it is within the map.
Definition map_func.h:316
static uint ScaleBySize(uint n)
Scales the given value by the map size, where the given value is for a 256 by 256 map.
Definition map_func.h:327
static uint SizeY()
Get the size of the map along the Y.
Definition map_func.h:278
static IterateWrapper Iterate()
Returns an iterable ensemble of all Tiles.
Definition map_func.h:362
static uint size
The number of tiles on the map.
Definition map_func.h:239
static debug_inline uint SizeX()
Get the size of the map along the X.
Definition map_func.h:269
static debug_inline uint LogX()
Logarithm of the map size along the X side.
Definition map_func.h:250
static uint size_x
Size of the map along the X.
Definition map_func.h:237
static uint tile_mask
_map_size - 1 (to mask the mapsize)
Definition map_func.h:240
static uint size_y
Size of the map along the Y.
Definition map_func.h:238
static uint ScaleBySize1D(uint n)
Scales the given value by the maps circumference, where the given value is for a 256 by 256 map.
Definition map_func.h:340
static uint LogY()
Logarithm of the map size along the y side.
Definition map_func.h:260
static bool IsInitialized()
Check whether the map has been initialized, as to not try to save the map during crashlog when the ma...
Definition map_func.h:353
static uint MaxY()
Gets the maximum Y coordinate within the map, including MP_VOID.
Definition map_func.h:305
static uint log_y
2^_map_log_y == _map_size_y
Definition map_func.h:236
static uint log_x
2^_map_log_x == _map_size_x
Definition map_func.h:235
static void Allocate(uint size_x, uint size_y)
(Re)allocates a map with the given dimension
Definition map.cpp:35
static debug_inline uint Size()
Get the size of the map.
Definition map_func.h:287
static debug_inline uint MaxX()
Gets the maximum X coordinate within the map, including MP_VOID.
Definition map_func.h:296
A pair-construct of a TileIndexDiff.
Definition map_type.h:31
int16_t x
The x value of the coordinate.
Definition map_type.h:32
int16_t y
The y value of the coordinate.
Definition map_type.h:33
Data that is stored per tile.
Definition map_func.h:32
uint8_t m3
General purpose.
Definition map_func.h:37
uint8_t m4
General purpose.
Definition map_func.h:38
uint8_t type
The type (bits 4..7), bridges (2..3), rainforest/desert (0..1)
Definition map_func.h:33
uint8_t m5
General purpose.
Definition map_func.h:39
uint8_t height
The height of the northern corner.
Definition map_func.h:34
uint8_t m1
Primarily used for ownership information.
Definition map_func.h:36
uint16_t m2
Primarily used for indices to towns, industries and stations.
Definition map_func.h:35
Data that is stored per tile.
Definition map_func.h:48
uint8_t m6
General purpose.
Definition map_func.h:49
uint8_t m7
Primarily used for newgrf support.
Definition map_func.h:50
uint16_t m8
General purpose.
Definition map_func.h:51
Types related to tiles.
constexpr TileIndex INVALID_TILE
The very nice invalid tile marker.
Definition tile_type.h:95
StrongType::Typedef< uint32_t, struct TileIndexTag, StrongType::Compare, StrongType::Integer, StrongType::Compatible< int32_t >, StrongType::Compatible< int64_t > > TileIndex
The index/ID of a Tile.
Definition tile_type.h:87