Как надежно определить Mac OS X, iOS, Linux, Windows in C препроцессор?

Если есть некоторый кроссплатформенный код C/C++, который должен быть скомпилирован в Mac OS X, iOS, Linux, Windows, как я могу надежно обнаружить их в процессе препроцессора?

3 ответа

Решение

Есть предопределенные макросы, которые используются большинством компиляторов, список вы можете найти здесь. Предопределенные макросы компилятора GCC можно найти здесь. Вот пример для gcc:

#ifdef _WIN32
   //define something for Windows (32-bit and 64-bit, this part is common)
   #ifdef _WIN64
      //define something for Windows (64-bit only)
   #else
      //define something for Windows (32-bit only)
   #endif
#elif __APPLE__
    #include "TargetConditionals.h"
    #if TARGET_IPHONE_SIMULATOR
         // iOS Simulator
    #elif TARGET_OS_IPHONE
        // iOS device
    #elif TARGET_OS_MAC
        // Other kinds of Mac OS
    #else
    #   error "Unknown Apple platform"
    #endif
#elif __linux__
    // linux
#elif __unix__ // all unices not caught above
    // Unix
#elif defined(_POSIX_VERSION)
    // POSIX
#else
#   error "Unknown compiler"
#endif

Определенные макросы зависят от компилятора, который вы собираетесь использовать.

_WIN64#ifdef может быть вложен в _WIN32#ifdef так как _WIN32 определяется при таргетинге на Windows, а не только на версию x86. Это предотвращает дублирование кода, если некоторые из них являются общими для обоих.

Как отмечает Джейк, TARGET_IPHONE_SIMULATOR является подмножеством TARGET_OS_IPHONE.

Кроме того, TARGET_OS_IPHONE является подмножеством TARGET_OS_MAC.

Таким образом, лучший подход может быть:

#ifdef _WIN64
   //define something for Windows (64-bit)
#elif _WIN32
   //define something for Windows (32-bit)
#elif __APPLE__
    #include "TargetConditionals.h"
    #if TARGET_OS_IPHONE && TARGET_IPHONE_SIMULATOR
        // define something for simulator   
    #elif TARGET_OS_IPHONE
        // define something for iphone  
    #else
        #define TARGET_OS_OSX 1
        // define something for OSX
    #endif
#elif __linux
    // linux
#elif __unix // all unices not caught above
    // Unix
#elif __posix
    // POSIX
#endif

Отчасти ответ: люди на [этом сайте] нашли время, чтобы составить таблицы макросов, определенные для каждой пары ОС / компилятор.

Например, вы можете видеть, что _WIN32 НЕ определяется в Windows с Cygwin (POSIX), в то время как он определен для компиляции в Windows, Cygwin (не-POSIX) и MinGW с каждым доступным компилятором (Clang, GNU, Intel и т. д.).

Во всяком случае, я нашел таблицы довольно информативными и решил поделиться здесь.

Другие вопросы по тегам