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/>.
8 /** @file stdafx.h Definition of base types and functions in a cross-platform compatible way. */
14 /* MinGW defaults to Windows 7 if none of these are set, and they must be set before any MinGW header is included */
15 # define NTDDI_VERSION NTDDI_WINXP // Windows XP
16 # define _WIN32_WINNT 0x501 // Windows XP
17 # define _WIN32_WINDOWS 0x501 // Windows XP
18 # define WINVER 0x0501 // Windows XP
19 # define _WIN32_IE_ 0x0600 // 6.0 (XP+)
23 /* Stop Microsoft (and clang-cl) compilers from complaining about potentially-unsafe/potentially-non-standard functions */
24 # define _CRT_SECURE_NO_DEPRECATE
25 # define _CRT_SECURE_NO_WARNINGS
26 # define _CRT_NONSTDC_NO_WARNINGS
29 #if defined(__APPLE__)
30 # include "os/macosx/osx_stdafx.h"
31 #endif /* __APPLE__ */
33 #if defined(__HAIKU__)
34 # include <SupportDefs.h>
36 # define _DEFAULT_SOURCE
38 # define TROUBLED_INTS
41 #if defined(__HAIKU__) || defined(__CYGWIN__)
42 # include <strings.h> /* strncasecmp */
45 /* It seems that we need to include stdint.h before anything else
46 * We need INT64_MAX, which for most systems comes from stdint.h. However, MSVC
47 * does not have stdint.h.
48 * For OSX the inclusion is already done in osx_stdafx.h. */
49 #if !defined(__APPLE__) && (!defined(_MSC_VER) || _MSC_VER >= 1600)
51 /* SunOS/Solaris does not have stdint.h, but inttypes.h defines everything
52 * stdint.h defines and we need. */
53 # include <inttypes.h>
55 # define __STDC_LIMIT_MACROS
60 /* The conditions for these constants to be available are way too messy; so check them one by one */
61 #if !defined(UINT64_MAX)
62 # define UINT64_MAX (18446744073709551615ULL)
64 #if !defined(INT64_MAX)
65 # define INT64_MAX (9223372036854775807LL)
67 #if !defined(INT64_MIN)
68 # define INT64_MIN (-INT64_MAX - 1)
70 #if !defined(UINT32_MAX)
71 # define UINT32_MAX (4294967295U)
73 #if !defined(INT32_MAX)
74 # define INT32_MAX (2147483647)
76 #if !defined(INT32_MIN)
77 # define INT32_MIN (-INT32_MAX - 1)
79 #if !defined(UINT16_MAX)
80 # define UINT16_MAX (65535U)
82 #if !defined(INT16_MAX)
83 # define INT16_MAX (32767)
85 #if !defined(INT16_MIN)
86 # define INT16_MIN (-INT16_MAX - 1)
88 #if !defined(UINT8_MAX)
89 # define UINT8_MAX (255)
91 #if !defined(INT8_MAX)
92 # define INT8_MAX (127)
94 #if !defined(INT8_MIN)
95 # define INT8_MIN (-INT8_MAX - 1)
109 # define SIZE_MAX ((size_t)-1)
112 #if defined(UNIX) || defined(__MINGW32__)
113 # include <sys/types.h>
118 # define strcasecmp stricmp
121 #if defined(SUNOS) || defined(HPUX) || defined(__CYGWIN__)
126 #if defined(__GNUC__) || (defined(__clang__) && !defined(_MSC_VER))
127 # define NORETURN __attribute__ ((noreturn))
129 # define __int64 long long
130 /* Warn about functions using 'printf' format syntax. First argument determines which parameter
131 * is the format string, second argument is start of values passed to printf. */
132 # define WARN_FORMAT(string, args) __attribute__ ((format (printf, string, args)))
133 # define WARN_TIME_FORMAT(string) __attribute__ ((format (strftime, string, 0)))
134 # if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 7)
140 /* Use fallthrough attribute where supported */
142 # if __cplusplus > 201402L // C++17
143 # define FALLTHROUGH [[fallthrough]]
145 # define FALLTHROUGH __attribute__((fallthrough))
150 #endif /* __GNUC__ || __clang__ */
152 #if __GNUC__ > 11 || (__GNUC__ == 11 && __GNUC_MINOR__ >= 1)
153 # define NOACCESS(args) __attribute__ ((access (none, args)))
155 # define NOACCESS(args)
158 #if defined(__WATCOMC__)
161 # define WARN_FORMAT(string, args)
162 # define WARN_TIME_FORMAT(string)
166 #endif /* __WATCOMC__ */
168 #if defined(__MINGW32__)
169 # include <malloc.h> // alloca()
173 # define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers
177 #if defined(_MSC_VER)
179 # define NOMINMAX // Disable min/max macros in windows.h.
181 # pragma warning(disable: 4244) // 'conversion' conversion from 'type1' to 'type2', possible loss of data
182 # pragma warning(disable: 4761) // integral size mismatch in argument : conversion supplied
183 # pragma warning(disable: 4200) // nonstandard extension used : zero-sized array in struct/union
184 # pragma warning(disable: 4355) // 'this' : used in base member initializer list
186 # if (_MSC_VER < 1400) // MSVC 2005 safety checks
187 # error "Only MSVC 2005 or higher are supported. MSVC 2003 and earlier are not! Upgrade your compiler."
188 # endif /* (_MSC_VER < 1400) */
189 # pragma warning(disable: 4291) // no matching operator delete found; memory will not be freed if initialization throws an exception (reason: our overloaded functions never throw an exception)
190 # pragma warning(disable: 4996) // 'function': was declared deprecated
191 # pragma warning(disable: 6308) // code analyzer: 'realloc' might return null pointer: assigning null pointer to 't_ptr', which is passed as an argument to 'realloc', will cause the original memory block to be leaked
192 # pragma warning(disable: 6011) // code analyzer: Dereferencing NULL pointer 'pfGetAddrInfo': Lines: 995, 996, 998, 999, 1001
193 # pragma warning(disable: 6326) // code analyzer: potential comparison of a constant with another constant
194 # pragma warning(disable: 6031) // code analyzer: Return value ignored: 'ReadFile'
195 # pragma warning(disable: 6255) // code analyzer: _alloca indicates failure by raising a stack overflow exception. Consider using _malloca instead
196 # pragma warning(disable: 6246) // code analyzer: Local declaration of 'statspec' hides declaration of the same name in outer scope. For additional information, see previous declaration at ...
198 # if (_MSC_VER == 1500) // Addresses item #13 on http://blogs.msdn.com/b/vcblog/archive/2008/08/11/tr1-fixes-in-vc9-sp1.aspx, for Visual Studio 2008
199 # define _DO_NOT_DECLARE_INTERLOCKED_INTRINSICS_IN_MEMORY
203 # include <malloc.h> // alloca()
204 # define NORETURN __declspec(noreturn)
205 # if (_MSC_VER < 1900)
206 # define inline __forceinline
209 # define CDECL _cdecl
210 # define WARN_FORMAT(string, args)
211 # define WARN_TIME_FORMAT(string)
214 /* fallthrough attribute, VS 2017 */
215 # if (_MSC_VER >= 1910) || defined(__clang__)
216 # define FALLTHROUGH [[fallthrough]]
221 # if defined(_WIN32) && !defined(_WIN64)
226 typedef _W64
int INT_PTR
, *PINT_PTR
;
227 typedef _W64
unsigned int UINT_PTR
, *PUINT_PTR
;
228 # endif /* _WIN32 && !_WIN64 */
231 # define fseek _fseeki64
234 /* zlib from vcpkg use cdecl calling convention without enforcing it in the headers */
235 # if defined(WITH_ZLIB)
236 # if !defined(ZEXPORT)
237 # define ZEXPORT CDECL
241 /* freetype from vcpkg use cdecl calling convention without enforcing it in the headers */
242 # if defined(WITH_FREETYPE)
243 # if !defined(FT_EXPORT)
244 # define FT_EXPORT( x ) extern "C" x CDECL
248 /* liblzma from vcpkg (before 5.2.4-2) used to patch lzma.h to define LZMA_API_STATIC for static builds */
249 # if defined(WITH_LIBLZMA)
250 # if !defined(LZMA_API_STATIC)
251 # define LZMA_API_STATIC
255 # define strcasecmp stricmp
256 # define strncasecmp strnicmp
257 # define strtoull _strtoui64
259 /* MSVC doesn't have these :( */
260 # define S_ISDIR(mode) (mode & S_IFDIR)
261 # define S_ISREG(mode) (mode & S_IFREG)
263 #endif /* defined(_MSC_VER) */
265 #if !defined(STRGEN) && !defined(SETTINGSGEN)
267 char *getcwd(char *buf
, size_t size
);
271 # define fopen(file, mode) _wfopen(OTTD2FS(file).c_str(), _T(mode))
272 # define unlink(file) _wunlink(OTTD2FS(file).c_str())
274 std::string
FS2OTTD(const std::wstring
&name
);
275 std::wstring
OTTD2FS(const std::string
&name
);
276 # elif defined(WITH_ICONV)
277 # define fopen(file, mode) fopen(OTTD2FS(file).c_str(), mode)
278 std::string
FS2OTTD(const std::string
&name
);
279 std::string
OTTD2FS(const std::string
&name
);
281 // no override of fopen() since no transformation is required of the filename
282 template <typename T
> std::string
FS2OTTD(T name
) { return name
; }
283 template <typename T
> std::string
OTTD2FS(T name
) { return name
; }
284 # endif /* _WIN32 or WITH_ICONV */
285 #endif /* STRGEN || SETTINGSGEN */
287 #if defined(_WIN32) || defined(__OS2__) && !defined(__INNOTEK_LIBC__)
288 # define PATHSEP "\\"
289 # define PATHSEPCHAR '\\'
292 # define PATHSEPCHAR '/'
295 #if defined(_MSC_VER) || defined(__WATCOMC__)
296 # define PACK_N(type_dec, n) __pragma(pack(push, n)) type_dec; __pragma(pack(pop))
297 #elif defined(__MINGW32__)
298 # define PRAGMA(x) _Pragma(#x)
299 # define PACK_N(type_dec, n) PRAGMA(pack(push, n)) type_dec; PRAGMA(pack(pop))
301 # define PACK_N(type_dec, n) type_dec __attribute__((__packed__, aligned(n)))
303 #define PACK(type_dec) PACK_N(type_dec, 1)
305 /* MSVCRT of course has to have a different syntax for long long *sigh* */
306 #if defined(_MSC_VER) || defined(__MINGW32__)
307 # define OTTD_PRINTF64 "%I64d"
308 # define OTTD_PRINTFHEX64 "%I64x"
309 # define PRINTF_SIZE "%Iu"
310 # define PRINTF_SIZEX "%IX"
312 # define OTTD_PRINTF64 "%lld"
313 # define OTTD_PRINTFHEX64 "%llx"
314 # define PRINTF_SIZE "%zu"
315 # define PRINTF_SIZEX "%zX"
318 typedef unsigned char byte
;
320 /* This is already defined in unix, but not in QNX Neutrino (6.x) or Cygwin. */
321 #if (!defined(UNIX) && !defined(__HAIKU__)) || defined(__QNXNTO__) || defined(__CYGWIN__)
322 typedef unsigned int uint
;
325 #if defined(TROUBLED_INTS)
326 /* Haiku's types for uint32/int32/uint64/int64 are different than what
327 * they are on other platforms; not in length, but how to print them.
328 * So make them more like the other platforms, to make printf() etc a
329 * little bit easier. */
330 # define uint32 uint32_ugly_hack
331 # define int32 int32_ugly_hack
332 # define uint64 uint64_ugly_hack
333 # define int64 int64_ugly_hack
334 typedef unsigned int uint32_ugly_hack
;
335 typedef signed int int32_ugly_hack
;
336 typedef unsigned __int64 uint64_ugly_hack
;
337 typedef signed __int64 int64_ugly_hack
;
339 typedef unsigned char uint8
;
340 typedef signed char int8
;
341 typedef unsigned short uint16
;
342 typedef signed short int16
;
343 typedef unsigned int uint32
;
344 typedef signed int int32
;
345 typedef unsigned __int64 uint64
;
346 typedef signed __int64 int64
;
347 #endif /* !TROUBLED_INTS */
349 #if !defined(WITH_PERSONAL_DIR)
350 # define PERSONAL_DIR ""
353 /* Define the the platforms that use XDG */
354 #if defined(WITH_PERSONAL_DIR) && defined(UNIX) && !defined(__APPLE__) && !defined(__EMSCRIPTEN__)
358 /* Check if the types have the bitsizes like we are using them */
359 static_assert(sizeof(uint64
) == 8);
360 static_assert(sizeof(uint32
) == 4);
361 static_assert(sizeof(uint16
) == 2);
362 static_assert(sizeof(uint8
) == 1);
363 static_assert(SIZE_MAX
>= UINT32_MAX
);
366 #define M_PI_2 1.57079632679489661923
367 #define M_PI 3.14159265358979323846
371 * Return the length of an fixed size array.
372 * Unlike sizeof this function returns the number of elements
375 * @param x The pointer to the first element of the array
376 * @return The number of elements
378 #define lengthof(x) (sizeof(x) / sizeof(x[0]))
381 * Get the end element of an fixed size array.
383 * @param x The pointer to the first element of the array
384 * @return The pointer past to the last element of the array
386 #define endof(x) (&x[lengthof(x)])
389 * Get the last element of an fixed size array.
391 * @param x The pointer to the first element of the array
392 * @return The pointer to the last element of the array
394 #define lastof(x) (&x[lengthof(x) - 1])
397 * Gets the size of a variable within a class.
398 * @param base The class the variable is in.
399 * @param variable The variable to get the size of.
400 * @return the size of the variable
402 #define cpp_sizeof(base, variable) (sizeof(std::declval<base>().variable))
405 * Gets the length of an array variable within a class.
406 * @param base The class the variable is in.
407 * @param variable The array variable to get the size of.
408 * @return the length of the array
410 #define cpp_lengthof(base, variable) (cpp_sizeof(base, variable) / cpp_sizeof(base, variable[0]))
413 /* take care of some name clashes on MacOS */
414 #if defined(__APPLE__)
415 # define GetString OTTD_GetString
416 # define DrawString OTTD_DrawString
417 # define CloseConnection OTTD_CloseConnection
418 #endif /* __APPLE__ */
420 #if defined(__GNUC__) || defined(__clang__)
421 # define likely(x) __builtin_expect(!!(x), 1)
422 # define unlikely(x) __builtin_expect(!!(x), 0)
424 # define likely(x) (x)
425 # define unlikely(x) (x)
426 #endif /* __GNUC__ || __clang__ */
428 /* For the FMT library we only want to use the headers, not link to some library. */
429 #define FMT_HEADER_ONLY
431 void NORETURN CDECL
usererror(const char *str
, ...) WARN_FORMAT(1, 2);
432 void NORETURN CDECL
error(const char *str
, ...) WARN_FORMAT(1, 2);
433 #define NOT_REACHED() error("NOT_REACHED triggered at line %i of %s", __LINE__, __FILE__)
435 /* For non-debug builds with assertions enabled use the special assertion handler. */
436 #if defined(NDEBUG) && defined(WITH_ASSERT)
438 # define assert(expression) if (unlikely(!(expression))) error("Assertion failed at line %i of %s: %s", __LINE__, __FILE__, #expression);
442 /* OpenBSD uses strcasecmp(3) */
443 # define _stricmp strcasecmp
446 #if defined(MAX_PATH)
447 /* It's already defined, no need to override */
448 #elif defined(PATH_MAX) && PATH_MAX > 0
449 /* Use the value from PATH_MAX, if it exists */
450 # define MAX_PATH PATH_MAX
452 /* If all else fails, hardcode something :( */
453 # define MAX_PATH 260
457 * Version of the standard free that accepts const pointers.
458 * @param ptr The data to free.
460 static inline void free(const void *ptr
)
462 free(const_cast<void *>(ptr
));
466 * The largest value that can be entered in a variable
467 * @param type the type of the variable
469 #define MAX_UVALUE(type) ((type)~(type)0)
471 #if defined(_MSC_VER) && !defined(_DEBUG)
472 # define IGNORE_UNINITIALIZED_WARNING_START __pragma(warning(push)) __pragma(warning(disable:4700))
473 # define IGNORE_UNINITIALIZED_WARNING_STOP __pragma(warning(pop))
474 #elif defined(__GNUC__) && !defined(_DEBUG)
475 # define HELPER0(x) #x
476 # define HELPER1(x) HELPER0(GCC diagnostic ignored x)
477 # define HELPER2(y) HELPER1(#y)
478 #if (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))
479 # define IGNORE_UNINITIALIZED_WARNING_START \
480 _Pragma("GCC diagnostic push") \
481 _Pragma(HELPER2(-Wuninitialized)) \
482 _Pragma(HELPER2(-Wmaybe-uninitialized))
483 # define IGNORE_UNINITIALIZED_WARNING_STOP _Pragma("GCC diagnostic pop")
487 #ifndef IGNORE_UNINITIALIZED_WARNING_START
488 # define IGNORE_UNINITIALIZED_WARNING_START
489 # define IGNORE_UNINITIALIZED_WARNING_STOP
492 #endif /* STDAFX_H */