blob: c3cc1a4a4afc8a3203d6db8197119680ffa1c684 [file] [log] [blame]
# Copyright 2019 The IREE Authors
#
# Licensed under the Apache License v2.0 with LLVM Exceptions.
# See https://llvm.org/LICENSE.txt for license information.
# SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
include(CMakeParseArguments)
# iree_bytecode_module()
#
# CMake function to imitate Bazel's iree_bytecode_module rule.
#
# Parameters:
# NAME: Name of target (see Note).
# SRC: Source file to compile into a bytecode module.
# FLAGS: Flags to pass to the translation tool (list of strings).
# TRANSLATE_TOOL: Translation tool to invoke (CMake target). The default
# tool is "iree-compile".
# C_IDENTIFIER: Identifier to use for generate c embed code.
# If omitted then no C embed code will be generated.
# PUBLIC: Add this so that this library will be exported under ${PACKAGE}::
# Also in IDE, target will appear in ${PACKAGE} folder while non PUBLIC
# will be in ${PACKAGE}/internal.
# TESTONLY: When added, this target will only be built if IREE_BUILD_TESTS=ON.
# MODULE_FILE_NAME: Optional. When specified, sets the output bytecode module
# file name. When not specified, a default file name will be generated from
# ${NAME}.
# DEPENDS: Optional. Additional dependencies beyond SRC and the tools.
# FRIENDLY_NAME: Optional. Name to use to display build progress info.
#
# Note:
# By default, iree_bytecode_module will create a library named ${NAME}_c,
# and alias target iree::${NAME}_c. The iree:: form should always be used.
# This is to reduce namespace pollution.
function(iree_bytecode_module)
cmake_parse_arguments(
_RULE
"PUBLIC;TESTONLY"
"NAME;SRC;TRANSLATE_TOOL;C_IDENTIFIER;MODULE_FILE_NAME;FRIENDLY_NAME"
"FLAGS;DEPENDS"
${ARGN}
)
if(_RULE_TESTONLY AND NOT IREE_BUILD_TESTS)
return()
endif()
# Set default for TRANSLATE_TOOL.
if(DEFINED _RULE_TRANSLATE_TOOL)
set(_TRANSLATE_TOOL ${_RULE_TRANSLATE_TOOL})
else()
set(_TRANSLATE_TOOL "iree-compile")
endif()
if(DEFINED _RULE_MODULE_FILE_NAME)
set(_MODULE_FILE_NAME "${_RULE_MODULE_FILE_NAME}")
else()
set(_MODULE_FILE_NAME "${_RULE_NAME}.vmfb")
endif()
iree_get_executable_path(_TRANSLATE_TOOL_EXECUTABLE ${_TRANSLATE_TOOL})
set(_ARGS "${_RULE_FLAGS}")
get_filename_component(_SRC_PATH "${_RULE_SRC}" REALPATH)
list(APPEND _ARGS "${_SRC_PATH}")
list(APPEND _ARGS "-o")
list(APPEND _ARGS "${_MODULE_FILE_NAME}")
# If an LLVM CPU backend is enabled, supply the linker tool.
if(IREE_LLD_TARGET)
iree_get_executable_path(_LINKER_TOOL_EXECUTABLE "lld")
list(APPEND _ARGS "-iree-llvm-embedded-linker-path=\"${_LINKER_TOOL_EXECUTABLE}\"")
list(APPEND _ARGS "-iree-llvm-wasm-linker-path=\"${_LINKER_TOOL_EXECUTABLE}\"")
# Note: -iree-llvm-system-linker-path is left unspecified.
endif()
if(IREE_BYTECODE_MODULE_FORCE_SYSTEM_DYLIB_LINKER)
list(APPEND _ARGS "-iree-llvm-link-embedded=false")
endif()
# Support testing in TSan build dirs. Unlike other sanitizers, TSan is an
# ABI break: when the host code is built with TSan, the module must be too,
# otherwise we get crashes calling module code.
if(IREE_BYTECODE_MODULE_ENABLE_TSAN)
list(APPEND _ARGS "-iree-llvm-sanitize=thread")
endif()
if(_RULE_FRIENDLY_NAME)
set(_FRIENDLY_NAME "${_RULE_FRIENDLY_NAME}")
else()
get_filename_component(_FRIENDLY_NAME "${_RULE_SRC}" NAME)
endif()
# Depending on the binary instead of the target here given we might not have
# a target in this CMake invocation when cross-compiling.
add_custom_command(
OUTPUT
"${_MODULE_FILE_NAME}"
COMMAND
${_TRANSLATE_TOOL_EXECUTABLE}
${_ARGS}
# Changes to either the translation tool or the input source should
# trigger rebuilding.
DEPENDS
${_TRANSLATE_TOOL_EXECUTABLE}
${_LINKER_TOOL_EXECUTABLE}
${_RULE_SRC}
${_RULE_DEPENDS}
COMMENT
"Generating VMFB for ${_FRIENDLY_NAME}"
VERBATIM
)
if(_RULE_TESTONLY)
set(_TESTONLY_ARG "TESTONLY")
endif()
if(_RULE_PUBLIC)
set(_PUBLIC_ARG "PUBLIC")
endif()
if(_RULE_C_IDENTIFIER)
iree_c_embed_data(
NAME
"${_RULE_NAME}_c"
IDENTIFIER
"${_RULE_C_IDENTIFIER}"
GENERATED_SRCS
"${_RULE_NAME}.vmfb"
C_FILE_OUTPUT
"${_RULE_NAME}_c.c"
H_FILE_OUTPUT
"${_RULE_NAME}_c.h"
FLATTEN
"${_PUBLIC_ARG}"
"${_TESTONLY_ARG}"
)
endif()
endfunction()