CMake modules to support compiling Apple Metal shaders as part of a CMake build system.
Find a file
2024-06-02 00:16:12 -07:00
cmake Match Xcode flags more closely 2024-06-01 23:53:20 -07:00
example Match Xcode flags more closely 2024-06-01 23:53:20 -07:00
.gitignore First commit of semi-broken code 2024-05-30 04:16:11 -07:00
CMakeLists.txt Apparently it's wrong to call enable_language before project 2024-05-31 18:21:25 -07:00
LICENCE.txt Licensing stuff 2024-05-30 20:02:13 -07:00
README.md README updates. Drop air/ar stuff because it doesn't work in Xcode 2024-06-02 00:16:12 -07:00

⚠️ INCOMPLETE! Metal Shader support for CMake

This project aims to allow compiling, linking, and embedding Apple Metal shader libraries in projects using CMake, without the need to write custom build steps or be restricted to only using the Xcode project generator.

Project Setup

  1. Copy the files from the cmake folder of this repo into a top-level cmake folder in your project.

  2. Add that cmake folder to the CMAKE_MODULE_PATH of your project:

    # In the top-level CMakeLists.txt file for your project:
    set(CMAKE_MODULE_PATH "${CMAKE_MODULE_PATH};${CMAKE_SOURCE_DIR}/cmake")
    
  3. Enable the Metal language for your project:

    enable_language(Metal)
    

    If you want to conditionally detect whether Metal can be supported, you can use check_language:

    include(CheckLanguage)
    
    check_language(Metal)
    if(CMAKE_Metal_COMPILER)
        enable_language(Metal)
    
        message(STATUS "Metal is supported")
    else()
        message(WARNING "Metal is not supported")
    endif()
    

Shader Library Setup

Important

Shaders must be in their own CMake target!
You cannot mix them with C++ or Objective-C code sources in a library or executable target.

Your final shader library will need to be a .metallib file that can be included in your application as a resource. You should define these as MODULE libraries rather than SHARED libraries because they do not expose symbols that can be linked with other targets.

add_library(shaders MODULE
    shader1.metal
    shader2.metal
)

# To make it work properly in Xcode with the Xcode project generator:
set_target_properties(shaders PROPERTIES
    XCODE_PRODUCT_TYPE com.apple.product-type.metal-library
    XCODE_ATTRIBUTE_MTL_FAST_MATH "YES"
    XCODE_ATTRIBUTE_MTL_ENABLE_DEBUG_INFO[variant=Debug] "INCLUDE_SOURCE"
    XCODE_ATTRIBUTE_MTL_ENABLE_DEBUG_INFO[variant=RelWithDebInfo] "INCLUDE_SOURCE"
)

Embedding Shaders

As a final step to make your shader library available to your application, you will need to ensure that it gets embedded as a resource.

  1. First, you need to ensure that the shader library itself is built as a dependency of the application target.

    add_executable(MyApp MACOSX_BUNDLE)
    
    add_dependencies(MyApp
        shaders
    )
    
  2. Then you'll need to include the shader libraries as resources:

    If you are using CMake 3.28 or newer with the Xcode generator, you can refer to the shader targets directly:

    set_target_properties(MyApp PROPERTIES
        XCODE_EMBED_RESOURCES shaders
    )
    

    Otherwise, you'll need a custom post-build step to copy the resulting .metallib into the Resources folder of the target app bundle.

    add_custom_command(TARGET MyApp POST_BUILD
        COMMAND ${CMAKE_COMMAND} -E copy "$<TARGET_FILE:shaders>" "$<TARGET_BUNDLE_CONTENT_DIR:MyApp>/Resources/$<TARGET_FILE_NAME:shaders>"
        VERBATIM
    )
    

Remaining Work

  • Handling the target and SDK parameters
  • Testing that this works with the Windows version of the Metal tools

Licence

CMake files are distributed under the OSI-approved BSD 3-clause License. See LICENCE.txt for details.

The example project is based on sample code from Apple, under its own licence. See LICENSE.txt in the example folder for details.