ScummVM API documentation
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
scummsys.h
1 /* ScummVM - Graphic Adventure Engine
2  *
3  * ScummVM is the legal property of its developers, whose names
4  * are too numerous to list here. Please refer to the COPYRIGHT
5  * file distributed with this source distribution.
6  *
7  * This program is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation, either version 3 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program. If not, see <http://www.gnu.org/licenses/>.
19  *
20  */
21 
22 #ifndef COMMON_SCUMMSYS_H
23 #define COMMON_SCUMMSYS_H
24 
25 #ifndef __has_feature // Optional of course.
26  #define __has_feature(x) 0 // Compatibility with non-clang compilers.
27 #endif
28 
29 // This is a convenience macro to test whether the compiler used is a GCC
30 // version, which is at least major.minor. Note that Clang will also define
31 // it and report itself as GCC 4.2.1.
32 #ifdef __GNUC__
33  #define GCC_ATLEAST(major, minor) (__GNUC__ > (major) || (__GNUC__ == (major) && __GNUC_MINOR__ >= (minor)))
34 #else
35  #define GCC_ATLEAST(major, minor) 0
36 #endif
37 
38 #if defined(NONSTANDARD_PORT)
39 
40  // Ports which need to perform #includes and #defines visible in
41  // virtually all the source of ScummVM should do so by providing a
42  // "portdefs.h" header file (and not by directly modifying this
43  // header file).
44  #include <portdefs.h>
45 #else // defined(NONSTANDARD_PORT)
46 
47  #if defined(WIN32)
48 
49  #if defined(_MSC_VER) && _MSC_VER <= 1800
50 
51  // FIXME: The placement of the workaround functions for MSVC below
52  // require us to include stdio.h and stdarg.h for MSVC here. This
53  // is not exactly nice...
54  // We should think of a better way of doing this.
55  #include <stdio.h>
56  #include <stdarg.h>
57 
58  // MSVC's vsnprintf is either non-existent (2003) or bugged since it
59  // does not always include a terminating NULL (2005+). To work around
60  // that we fix up the _vsnprintf included. Note that the return value
61  // will still not match C99's specs!
62  inline int vsnprintf_msvc(char *str, size_t size, const char *format, va_list args) {
63  // We do not pass size - 1 here, to ensure we would get the same
64  // return value as when we would use _vsnprintf directly, since
65  // for example Common::String::format relies on this.
66  int retValue = _vsnprintf(str, size, format, args);
67  str[size - 1] = 0;
68  return retValue;
69  }
70 
71  #define vsnprintf vsnprintf_msvc
72 
73  // Visual Studio does not include snprintf in its standard C library.
74  // Instead it includes a function called _snprintf with somewhat
75  // similar semantics. The minor difference is that the return value in
76  // case the formatted string exceeds the buffer size is different.
77  // A much more dangerous one is that _snprintf does not always include
78  // a terminating null (Whoops!). Instead we map to our fixed vsnprintf.
79  inline int snprintf(char *str, size_t size, const char *format, ...) {
80  va_list args;
81  va_start(args, format);
82  int len = vsnprintf(str, size, format, args);
83  va_end(args);
84  return len;
85  }
86  #endif
87 
88  #define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers
89  #define NOGDICAPMASKS
90  #define OEMRESOURCE
91  #define NONLS
92  #define NOICONS
93  #define NOMCX
94  #define NOPROFILER
95  #define NOKANJI
96  #define NOSERVICE
97  #define NOMETAFILE
98  #define NOCOMM
99  #define NOCRYPT
100  #define NOIME
101  #define NOATOM
102  #define NOCTLMGR
103  #define NOCLIPBOARD
104  #define NOMEMMGR
105  #define NOSYSMETRICS
106  #define NOMENUS
107  #define NOOPENFILE
108  #define NOWH
109  #define NOSOUND
110  #define NODRAWTEXT
111  #define NOMINMAX 1
112 
113  #endif
114 
115  #if defined(__QNXNTO__)
116  #include <strings.h> /* For strcasecmp */
117  #endif
118 
119  #include <stdio.h>
120  #include <stdlib.h>
121  #include <string.h>
122  #include <stdarg.h>
123  #include <stddef.h>
124  #include <assert.h>
125  #include <ctype.h>
126 
127  // The C++11 standard removed the C99 requirement that some <inttypes.h>
128  // features should only be available when the following macros are defined.
129  // But on some systems (such as RISC OS or macOS < 10.7), the system headers
130  // are not necessarily up to date with this change. So, continue defining
131  // them, in order to avoid build failures on some environments.
132  #define __STDC_CONSTANT_MACROS
133  #define __STDC_FORMAT_MACROS
134  #define __STDC_LIMIT_MACROS
135  #include <inttypes.h>
136 
137  // macOS 10.4 inttypes.h woes -- fixed in 10.5 SDK
138  #if defined(MACOSX) && defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) && __ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__ < 1050
139  #undef __PRI_8_LENGTH_MODIFIER__
140  #undef __PRI_64_LENGTH_MODIFIER__
141  #undef __SCN_64_LENGTH_MODIFIER__
142  #define __PRI_8_LENGTH_MODIFIER__ "hh"
143  #define __PRI_64_LENGTH_MODIFIER__ "ll"
144  #define __SCN_64_LENGTH_MODIFIER__ "ll"
145  #endif
146 
147  #include <limits.h>
148  // MSVC does not define M_PI, M_SQRT2 and other math defines by default.
149  // _USE_MATH_DEFINES must be defined in order to have these defined, thus
150  // we enable it here. For more information, check:
151  // https://docs.microsoft.com/en-us/cpp/c-runtime-library/math-constants
152  #define _USE_MATH_DEFINES
153  #include <math.h>
154 
155  // FIXME: We sadly can't assume standard C++ headers to be present on every
156  // system we support, so we should get rid of this. The solution should be to
157  // write a simple placement new on our own. It might be noteworthy we can't
158  // easily do that for systems which do have a <new>, since it might clash with
159  // the default definition otherwise!
160  #include <new>
161 
162  // After having discussion about use of std::numeric_limits in
163  // https://github.com/scummvm/scummvm/pull/3966 we concluded that it is safe to
164  // use it as minimal STL code with low probability to bring any incompatibilities
165  #include <limits>
166 #endif
167 
168 #ifndef STATIC_ASSERT
169 #if __cplusplus >= 201103L || (defined(_MSC_VER) && _MSC_VER > 1600)
170 
177  #define STATIC_ASSERT(expression, message) \
178  static_assert((expression), #message)
179 #else
180 
187  #define STATIC_ASSERT(expression, message) \
188  do { \
189  extern int STATIC_ASSERT_##message[(expression) ? 1 : -1]; \
190  (void)(STATIC_ASSERT_##message); \
191  } while (false)
192 #endif
193 #endif
194 
195 // The following math constants are usually defined by the system math.h header, but
196 // they are not part of the ANSI C++ standards and so can NOT be relied upon to be
197 // present i.e. when -std=c++11 is passed to GCC, enabling strict ANSI compliance.
198 // As we rely on these being present, we define them if they are not set.
199 
200 #ifndef M_E
201  #define M_E 2.7182818284590452354 /* e */
202 #endif
203 
204 #ifndef M_LOG2E
205  #define M_LOG2E 1.4426950408889634074 /* log_2 e */
206 #endif
207 
208 #ifndef M_LOG10E
209  #define M_LOG10E 0.43429448190325182765 /* log_10 e */
210 #endif
211 
212 #ifndef M_LN2
213  #define M_LN2 0.69314718055994530942 /* log_e 2 */
214 #endif
215 
216 #ifndef M_LN10
217  #define M_LN10 2.30258509299404568402 /* log_e 10 */
218 #endif
219 
220 #ifndef M_PI
221  #define M_PI 3.14159265358979323846 /* pi */
222 #endif
223 
224 #ifndef M_PI_2
225  #define M_PI_2 1.57079632679489661923 /* pi/2 */
226 #endif
227 
228 #ifndef M_PI_4
229  #define M_PI_4 0.78539816339744830962 /* pi/4 */
230 #endif
231 
232 #ifndef M_1_PI
233  #define M_1_PI 0.31830988618379067154 /* 1/pi */
234 #endif
235 
236 #ifndef M_2_PI
237  #define M_2_PI 0.63661977236758134308 /* 2/pi */
238 #endif
239 
240 #ifndef M_2_SQRTPI
241  #define M_2_SQRTPI 1.12837916709551257390 /* 2/sqrt(pi) */
242 #endif
243 
244 #ifndef M_SQRT2
245  #define M_SQRT2 1.41421356237309504880 /* sqrt(2) */
246 #endif
247 
248 #ifndef M_SQRT1_2
249  #define M_SQRT1_2 0.70710678118654752440 /* 1/sqrt(2) */
250 #endif
251 
252 // Use config.h, generated by configure
253 #if defined(HAVE_CONFIG_H)
254 #include "config.h"
255 #endif
256 
257 
258 // In the following we configure various targets, in particular those
259 // which can't use our "configure" tool and hence don't use config.h.
260 //
261 // Some #defines that occur here frequently:
262 // SCUMM_LITTLE_ENDIAN
263 // - Define this on a little endian target
264 // SCUMM_BIG_ENDIAN
265 // - Define this on a big endian target
266 // SCUMM_NEED_ALIGNMENT
267 // - Define this if your system has problems reading e.g. an int32 from an odd address
268 // SMALL_SCREEN_DEVICE
269 // - ...
270 // ...
271 
272 
273 //
274 // By default we try to use pragma push/pop to ensure various structs we use
275 // are "packed". If your compiler doesn't support this pragma, you are in for
276 // a problem. If you are lucky, there is a compiler switch, or another pragma,
277 // doing the same thing -- in that case, try to modify common/pack-begin.h and
278 // common/pack-end.h accordingly. Or maybe your port simply *always* packs
279 // everything, in which case you could #undefine SCUMMVM_USE_PRAGMA_PACK.
280 //
281 // If neither is possible, tough luck. Try to contact the team, maybe we can
282 // come up with a solution, though I wouldn't hold my breath on it :-/.
283 //
284 #define SCUMMVM_USE_PRAGMA_PACK
285 
286 //
287 // Determine the host endianess and whether memory alignment is required.
288 //
289 #if !defined(HAVE_CONFIG_H)
290 
291  // If -fsanitize=undefined or -fsanitize=alignment is in use, and the
292  // compiler happens to report it, make sure SCUMM_NEED_ALIGNMENT is
293  // defined, in order to avoid false positives when not using the
294  // "configure" script to enable UBSan.
295  #if __has_feature(undefined_behavior_sanitizer)
296  #define SCUMM_NEED_ALIGNMENT
297  #endif
298 
299  #if defined(__DC__) || \
300  defined(__DS__) || \
301  defined(__3DS__) || \
302  defined(IPHONE) || \
303  defined(__PSP__)
304 
305  #define SCUMM_LITTLE_ENDIAN
306  #define SCUMM_NEED_ALIGNMENT
307 
308  #elif defined(_MSC_VER) || defined(__MINGW32__)
309 
310  #define SCUMM_LITTLE_ENDIAN
311 
312  #elif defined(__MORPHOS__) || defined(__amigaos4__) || defined(__N64__) || defined(__WII__)
313 
314  #define SCUMM_BIG_ENDIAN
315  #define SCUMM_NEED_ALIGNMENT
316 
317  #elif defined(SDL_BACKEND)
318  // On SDL based ports, we try to use SDL_BYTEORDER to determine the
319  // endianess. We explicitly do this as the *last* thing we try, so that
320  // platform specific settings have precedence.
321  #include <SDL_endian.h>
322 
323  #if SDL_BYTEORDER == SDL_LIL_ENDIAN
324  #define SCUMM_LITTLE_ENDIAN
325  #elif SDL_BYTEORDER == SDL_BIG_ENDIAN
326  #define SCUMM_BIG_ENDIAN
327  #else
328  #error Neither SDL_BIG_ENDIAN nor SDL_LIL_ENDIAN is set.
329  #endif
330 
331  #else
332 
333  #error No system type defined, host endianess unknown.
334 
335  #endif
336 #endif
337 
338 #if !defined(SCUMM_FLOAT_WORD_LITTLE_ENDIAN) && !defined(SCUMM_FLOAT_WORD_BIG_ENDIAN)
339 
340  #if defined(__ORDER_LITTLE_ENDIAN__) && defined(__ORDER_BIG_ENDIAN__) && defined(__FLOAT_WORD_ORDER__)
341 
342  #if (__FLOAT_WORD_ORDER__ == __ORDER_LITTLE_ENDIAN__)
343  #define SCUMM_FLOAT_WORD_LITTLE_ENDIAN
344  #elif (__FLOAT_WORD_ORDER__ == __ORDER_BIG_ENDIAN__)
345  #define SCUMM_FLOAT_WORD_BIG_ENDIAN
346  #else
347  #error Unsupported endianness
348  #endif
349 
350  #else
351  #ifdef SCUMM_LITTLE_ENDIAN
352  #define SCUMM_FLOAT_WORD_LITTLE_ENDIAN
353  #else
354  #define SCUMM_FLOAT_WORD_BIG_ENDIAN
355  #endif
356 
357  #endif
358 #endif
359 
360 #if defined(USE_TREMOR) && !defined(USE_VORBIS)
361 #define USE_VORBIS // make sure this one is defined together with USE_TREMOR!
362 #endif
363 
364 //
365 // Fallbacks / default values for various special macros
366 //
367 #ifndef GCC_PRINTF
368  #if defined(__GNUC__) || defined(__INTEL_COMPILER)
369  #if defined(__USE_MINGW_ANSI_STDIO) && __USE_MINGW_ANSI_STDIO && !defined(__clang__)
370  #define GCC_PRINTF(x,y) __attribute__((__format__(__gnu_printf__, x, y)))
371  #else
372  #define GCC_PRINTF(x,y) __attribute__((__format__(__printf__, x, y)))
373  #endif
374  #else
375  #define GCC_PRINTF(x,y)
376  #endif
377 #endif
378 
379 #ifndef MSVC_PRINTF
380  #if defined(_MSC_VER) && _MSC_VER > 1400
381  #define MSVC_PRINTF _Printf_format_string_
382  #else
383  #define MSVC_PRINTF
384  #endif
385 #endif
386 
387 #ifndef PACKED_STRUCT
388  #if defined(__GNUC__) || defined(__INTEL_COMPILER)
389  #define PACKED_STRUCT __attribute__((__packed__))
390  #else
391  #define PACKED_STRUCT
392  #endif
393 #endif
394 
395 #ifndef FORCEINLINE
396  #if defined(_MSC_VER)
397  #define FORCEINLINE __forceinline
398  #elif defined(__GNUC__)
399  #define FORCEINLINE inline __attribute__((__always_inline__))
400  #else
401  #define FORCEINLINE inline
402  #endif
403 #endif
404 
405 #ifndef PLUGIN_EXPORT
406  #if defined(_MSC_VER) || defined(__MINGW32__)
407  #define PLUGIN_EXPORT __declspec(dllexport)
408  #else
409  #define PLUGIN_EXPORT
410  #endif
411 #endif
412 
413 #ifndef NORETURN_PRE
414  #if defined(_MSC_VER)
415  #define NORETURN_PRE __declspec(noreturn)
416  #elif defined(__GNUC__)
417  #define NORETURN_PRE __attribute__((__noreturn__))
418  #else
419  #define NORETURN_PRE
420  #endif
421 #endif
422 
423 #ifndef NORETURN_POST
424  #if defined(__INTEL_COMPILER)
425  #define NORETURN_POST __attribute__((__noreturn__))
426  #else
427  #define NORETURN_POST
428  #endif
429 #endif
430 
431 #ifndef WARN_UNUSED_RESULT
432  #if __cplusplus >= 201703L
433  #define WARN_UNUSED_RESULT [[nodiscard]]
434  #elif defined(__GNUC__)
435  #define WARN_UNUSED_RESULT __attribute__((__warn_unused_result__))
436  #elif defined(_Check_return_)
437  #define WARN_UNUSED_RESULT _Check_return_
438  #else
439  #define WARN_UNUSED_RESULT
440  #endif
441 #endif
442 
443 #ifndef WARN_DEPRECATED
444  #if __cplusplus >= 201703L
445  #define WARN_DEPRECATED(msg) [[deprecated(msg)]]
446  #elif defined(__GNUC__)
447  #define WARN_DEPRECATED(msg) __attribute__((__deprecated__(msg)))
448  #elif defined(_MSC_VER)
449  #define WARN_DEPRECATED(msg) __declspec(deprecated(msg))
450  #else
451  #define WARN_DEPRECATED(msg)
452  #endif
453 #endif
454 
455 #ifndef STRINGBUFLEN
456  #if defined(__N64__) || defined(__DS__) || defined(__3DS__)
457  #define STRINGBUFLEN 256
458  #else
459  #define STRINGBUFLEN 1024
460  #endif
461 #endif
462 
463 #ifndef MAXPATHLEN
464 #define MAXPATHLEN 256
465 #endif
466 
467 #ifndef scumm_va_copy
468  #if defined(va_copy)
469  #define scumm_va_copy va_copy
470  #elif defined(__va_copy)
471  #define scumm_va_copy __va_copy
472  #elif defined(_MSC_VER)
473  #define scumm_va_copy(dst, src) ((dst) = (src))
474  #else
475  #error scumm_va_copy undefined for this port
476  #endif
477 #endif
478 
479 
480 
481 //
482 // Typedef our system types unless SCUMMVM_DONT_DEFINE_TYPES is set.
483 // This is usually due to conflicts with the system headers.
484 //
485 #ifndef SCUMMVM_DONT_DEFINE_TYPES
486  typedef unsigned char byte;
487  typedef unsigned int uint;
488 
489  typedef uint8_t uint8;
490  typedef int8_t int8;
491  typedef uint16_t uint16;
492  typedef int16_t int16;
493 #if defined(__3DS__) || defined(__DC__) || defined(__PSP__)
494 
498  typedef unsigned int uint32;
499  typedef int int32;
500 #elif defined(__amigaos4__) || defined(__MORPHOS__)
501 
506  typedef unsigned long uint32;
507  typedef signed long int32;
508 #else
509  typedef uint32_t uint32;
510  typedef int32_t int32;
511 #endif
512  typedef uint64_t uint64;
513  typedef int64_t int64;
514  typedef uintptr_t uintptr;
515  typedef intptr_t intptr;
516 #endif
517 
518 //
519 // std::nullptr_t when this type is not available
520 //
521 #if defined(NO_CXX11_NULLPTR_T)
522 namespace std {
523  typedef decltype(nullptr) nullptr_t;
524 }
525 #endif
526 
527 //
528 // std::initializer_list
529 // Provide replacement when not available
530 //
531 #if defined(NO_CXX11_INITIALIZER_LIST)
532 namespace std {
533  template<class T> class initializer_list {
534  public:
535  typedef T value_type;
536  typedef const T &reference;
537  typedef const T &const_reference;
538  typedef size_t size_type;
539  typedef const T *iterator;
540  typedef const T *const_iterator;
541 
542  constexpr initializer_list() noexcept = default;
543  constexpr size_t size() const noexcept { return _size; };
544  constexpr const T *begin() const noexcept { return _begin; };
545  constexpr const T *end() const noexcept { return _begin + _size; }
546 
547  private:
548  // Note: begin has to be first or the compiler may get very upset
549  const T *_begin = { nullptr };
550  size_t _size = { 0 };
551 
552  // The compiler is allowed to call this constructor
553  constexpr initializer_list(const T* t, size_t s) noexcept : _begin(t) , _size(s) {}
554  };
555 
556  template<class T> constexpr const T* begin(initializer_list<T> il) noexcept {
557  return il.begin();
558  }
559 
560  template<class T> constexpr const T* end(initializer_list<T> il) noexcept {
561  return il.end();
562  }
563 }
564 
565 #else
566 
567 #include <initializer_list>
568 
569 #endif // NO_CXX11_INITIALIZER_LIST
570 
571 //
572 // Some more system specific settings.
573 // TODO/FIXME: All of these should be moved to backend specific files (such as portdefs.h)
574 //
575 #if defined(DINGUX)
576 
577  // Very BAD hack following, used to avoid triggering an assert in uClibc dingux library
578  // "toupper" when pressing keyboard function keys.
579  #undef toupper
580  #define toupper(c) __extension__ ({ auto _x = ((c) & 0xFF); (_x >= 97 && _x <= 122) ? (_x - 32) : _x; })
581 
582 #elif defined(__PSP__)
583 
584  #include <malloc.h>
585  #include "backends/platform/psp/memory.h"
586 
587  /* to make an efficient, inlined memcpy implementation */
588  #define memcpy(dst, src, size) psp_memcpy(dst, src, size)
589 
590 #endif
591 
592 #include "common/forbidden.h"
593 
594 #endif
Definition: algorithm.h:37