Подтвердить что ты не робот

Как настроить CMake для создания приложения для iPhone

Это тесно связано с предыдущим вопросом , который касался использования CMake для создания статической библиотеки на iPhone. Я получил это для настройки CMAKE_OSX_SYSROOT.

Однако это не работает для создания приложения. Мой CMakeLists.txt выглядит так:

project(TEST)

set(CMAKE_OSX_SYSROOT iphoneos2.2.1)
set(CMAKE_OSX_ARCHITECTURES "$(ARCHS_STANDARD_32_BIT)")
set(CMAKE_EXE_LINKER_FLAGS
"-framework Foundation -framework OpenGLES -framework AudioToolbox -framework CoreGraphics -framework QuartzCore -framework UIKit -framework OpenAL"
)

set(SRC --my files--)

add_executable(iphone-test MACOSX_BUNDLE ${SRC})

Несколько примечаний:

  • Я явно предоставляю ссылку -framework, потому что find_library не работает для всех фреймворков (он нашел большинство из них, но не OpenGLES). Я не понимаю, почему, поскольку все они находятся в одной папке ${SDK}/System/Library/Frameworks. Это заставляет меня думать, что я делаю что-то неправильно, но я не знаю, что.
  • Я добавил MACOSX_BUNDLE в команду add_executable, чтобы сгенерированный тип продукта был com.apple.product-type.application вместо com.apple.product-type.tool, который, по-видимому, не существует на iPhone.

В любом случае приложение компилируется и правильно связывается, но когда я запускаю его в симуляторе, я получаю страшный

Failed to launch simulated application: Unknown error.

Существует множество сообщений об этой проблеме в google и stackoverflow, но все решения включают очистку или создание нового проекта и перемещение файлов; но я компилирую новую копию после того, как CMake выполняет свою работу, поэтому ничего из этого не имеет.

Я нашел этот поток в списке рассылки CMake, но он только сообщает об успехе в создании библиотеки, а затем peters out.

4b9b3361

Ответ 1

Наконец-то я понял, как это сделать. Вот как выглядит мой файл CMakeLists.txt:

project(test)
set(NAME test)

file(GLOB headers *.h)
file(GLOB sources *.cpp)

set(CMAKE_OSX_SYSROOT iphoneos2.2.1)
set(CMAKE_OSX_ARCHITECTURES $(ARCHS_STANDARD_32_BIT))
set(CMAKE_CXX_FLAGS "-x objective-c++")
set(CMAKE_EXE_LINKER_FLAGS
    "-framework AudioToolbox -framework CoreGraphics -framework QuartzCore -framework UIKit"
)
link_directories(\${HOME}/\${SDKROOT}/lib)

set(MACOSX_BUNDLE_GUI_IDENTIFIER "com.mycompany.\${PRODUCT_NAME:identifier}")
set(APP_TYPE MACOSX_BUNDLE)

add_executable(${NAME}
    ${APP_TYPE}
    ${headers}
    ${sources}
)

target_link_libraries(${NAME}
    # other libraries to link
)

# code signing
set_target_properties(${NAME} PROPERTIES XCODE_ATTRIBUTE_CODE_SIGN_IDENTITY "iPhone Developer: My Name")

Очевидно, измените mycompany на название вашей компании и измените My Name на свое имя. Мне было очень полезно добавить каталог ссылок \${HOME}/\${SDKROOT}/lib, как указано выше, так что если ваше приложение ссылается на статическую библиотеку (особенно на общую (не iPhone) библиотеку), вы можете создавать отдельные библиотеки iPhoneOS и iPhoneSimulator и легко ссылаться на правильный, вместо того, чтобы беспокоиться о универсальном двоичном файле.

Кроме того, Xcode, похоже, не правильно добавляет ресурсы при создании проекта с использованием CMake, поэтому я добавил эту часть в файл CMakeLists.txt. Он копирует всю папку /data в папку моих ресурсов (как если бы я добавил ссылку на синюю папку в Xcode).

# copy resource phase

set(APP_NAME \${TARGET_BUILD_DIR}/\${FULL_PRODUCT_NAME})
set(RES_DIR ${test_SOURCE_DIR}/data)
add_custom_command(
    TARGET ${NAME}
    POST_BUILD
    COMMAND /Developer/Library/PrivateFrameworks/DevToolsCore.framework/Resources/pbxcp -exclude .DS_Store -exclude CVS -exclude .svn -resolve-src-symlinks ${RES_DIR} ${APP_NAME}
)

Ответ 2

Для фреймворков я нашел это сообщение http://www.mail-archive.com/[email protected]/msg24659.html Я хватал достаточно информации для этого:

SET(TARGETSDK iPhoneOS3.1.2.sdk)
SET(CMAKE_OSX_SYSROOT /Developer/Platforms/iPhoneOS.platform/Developer/SDKs/${TARGETSDK})
macro(ADD_FRAMEWORK fwname appname)
    find_library(FRAMEWORK_${fwname}
        NAMES ${fwname}
        PATHS ${CMAKE_OSX_SYSROOT}/System/Library
        PATH_SUFFIXES Frameworks
        NO_DEFAULT_PATH)
    if( ${FRAMEWORK_${fwname}} STREQUAL FRAMEWORK_${fwname}-NOTFOUND)
        MESSAGE(ERROR ": Framework ${fwname} not found")
    else()
        TARGET_LINK_LIBRARIES(${appname} ${FRAMEWORK_${fwname}})
        MESSAGE(STATUS "Framework ${fwname} found at ${FRAMEWORK_${fwname}}")
    endif()
endmacro(ADD_FRAMEWORK)

Вместо того, чтобы устанавливать CMAKE_EXE_LINKER_FLAGS, теперь я делаю:

ADD_FRAMEWORK(AudioToolbox MyApp)
ADD_FRAMEWORK(CoreGraphics MyApp)
ADD_FRAMEWORK(QuartzCore MyApp)
ADD_FRAMEWORK(UIKit MyApp)

Это также работает для OpenGLES.

Ответ 3

Последние версии CMake передают файлы .m и .mm в компилятор С++, который обнаруживает язык через расширение, поэтому вам не нужно явно добавлять "-x objective-c ++" в флаги.