include(FindJava) find_package(Java REQUIRED) include(UseJava) if (NOT ANDROID) set(JAVA_AWT_LIBRARY NotNeeded) set(JAVA_JVM_LIBRARY NotNeeded) set(JAVA_INCLUDE_PATH2 NotNeeded) set(JAVA_AWT_INCLUDE_PATH NotNeeded) find_package(JNI REQUIRED) endif() set(JAVA_ROOT ${PROJECT_SOURCE_DIR}/java) set(JAVA_OUTPUT_TEMP ${CMAKE_CURRENT_BINARY_DIR}/java-temp) set(JAVA_OUTPUT_DIR ${CMAKE_CURRENT_BINARY_DIR}/java) # use the gradle wrapper if it exists if(EXISTS "${JAVA_ROOT}/gradlew") set(GRADLE_EXECUTABLE "${JAVA_ROOT}/gradlew") else() # fall back to gradle on our PATH find_program(GRADLE_EXECUTABLE gradle) if(NOT GRADLE_EXECUTABLE) message(SEND_ERROR "Gradle installation not found") endif() endif() message(STATUS "Using gradle: ${GRADLE_EXECUTABLE}") # Specify the Java source files file(GLOB_RECURSE onnxruntime_extensions4j_gradle_files "${JAVA_ROOT}/*.gradle") file(GLOB_RECURSE onnxruntime_extensions4j_src "${JAVA_ROOT}/src/main/java/ai/onnxruntime/extensions/*.java") set(JAVA_OUTPUT_JAR ${JAVA_OUTPUT_TEMP}/build/libs/onnxruntime_extensions.jar) # this jar is solely used to signaling mechanism for dependency management in CMake # if any of the Java sources change, the jar (and generated headers) will be regenerated # and the onnxruntime_extensions4j_jni target will be rebuilt set(GRADLE_ARGS --console=plain clean jar -p ${JAVA_ROOT} -x test ) if(WIN32) set(GRADLE_ARGS ${GRADLE_ARGS} -Dorg.gradle.daemon=false) elseif (ANDROID) # For Android build, we may run gradle multiple times in same build, # sometimes gradle JVM will run out of memory if we keep the daemon running # it is better to not keep a daemon running set(GRADLE_ARGS ${GRADLE_ARGS} --no-daemon) endif() file(MAKE_DIRECTORY ${JAVA_OUTPUT_TEMP}) add_custom_command(OUTPUT ${JAVA_OUTPUT_JAR} COMMAND ${GRADLE_EXECUTABLE} ${GRADLE_ARGS} WORKING_DIRECTORY ${JAVA_OUTPUT_TEMP} DEPENDS ${onnxruntime_extensions4j_gradle_files} ${onnxruntime_extensions4j_src} ortcustomops) add_custom_target(onnxruntime_extensions4j DEPENDS ${JAVA_OUTPUT_JAR}) set_source_files_properties(${JAVA_OUTPUT_JAR} PROPERTIES GENERATED TRUE) set_property(TARGET onnxruntime_extensions4j APPEND PROPERTY ADDITIONAL_CLEAN_FILES "${JAVA_OUTPUT_DIR}") # Specify the native sources file(GLOB onnxruntime_extensions4j_native_src "${JAVA_ROOT}/src/main/native/*.c" "${JAVA_ROOT}/src/main/native/*.h" "${PROJECT_SOURCE_DIR}/include/*.h" ) # Build the JNI library add_library(onnxruntime_extensions4j_jni SHARED ${onnxruntime_extensions4j_native_src}) # depend on java sources. if they change, the JNI should recompile add_dependencies(onnxruntime_extensions4j_jni onnxruntime_extensions4j) target_include_directories(onnxruntime_extensions4j_jni PRIVATE ortcustomops) # the JNI headers are generated in the onnxruntime_extensions4j target target_include_directories(onnxruntime_extensions4j_jni PRIVATE ${JAVA_ROOT}/build/headers ${JNI_INCLUDE_DIRS}) # use shared lib for extensions on Android as NuGet requires the extensions .so if (ANDROID AND _BUILD_SHARED_LIBRARY) target_link_libraries(onnxruntime_extensions4j_jni PRIVATE extensions_shared) else() target_link_libraries(onnxruntime_extensions4j_jni PRIVATE ortcustomops) endif() standardize_output_folder(onnxruntime_extensions4j_jni) # Set platform and arch for packaging # Checks the names set by MLAS on non-Windows platforms first if(APPLE) get_target_property(ONNXRUNTIME4J_OSX_ARCH onnxruntime_extensions4j_jni OSX_ARCHITECTURES) list(LENGTH ONNXRUNTIME4J_OSX_ARCH ONNXRUNTIME4J_OSX_ARCH_LEN) if(ONNXRUNTIME4J_OSX_ARCH) if(ONNXRUNTIME4J_OSX_ARCH_LEN LESS_EQUAL 1) list(GET ONNXRUNTIME4J_OSX_ARCH 0 JNI_ARCH) message("Set Java ARCH TO macOS/iOS ${JNI_ARCH}") else() message(FATAL_ERROR "Java is currently not supported for macOS universal") endif() else() set(JNI_ARCH ${CMAKE_HOST_SYSTEM_PROCESSOR}) message("Set Java ARCH TO macOS/iOS ${JNI_ARCH}") endif() if(JNI_ARCH STREQUAL "x86_64") set(JNI_ARCH x64) elseif(JNI_ARCH STREQUAL "arm64") set(JNI_ARCH aarch64) endif() elseif (ANDROID) set(JNI_ARCH ${ANDROID_ABI}) elseif (CMAKE_SYSTEM_PROCESSOR STREQUAL "x86_64") set(JNI_ARCH x64) else() # Now mirror the checks used with MSVC if(MSVC) if(CMAKE_GENERATOR_PLATFORM STREQUAL "ARM64") set(JNI_ARCH aarch64) elseif(CMAKE_GENERATOR_PLATFORM STREQUAL "x64") set(JNI_ARCH x64) else() # if everything else failed then we're on a 32-bit arch and Java isn't supported message(FATAL_ERROR "Java is currently not supported on 32-bit x86 architecture") endif() else() # if everything else failed then we're on a 32-bit arch and Java isn't supported message(FATAL_ERROR "Java is currently not supported on 32-bit x86 architecture") endif() endif() if (WIN32) set(JAVA_PLAT "win") elseif (APPLE) set(JAVA_PLAT "osx") elseif (${CMAKE_SYSTEM_NAME} MATCHES "Linux") set(JAVA_PLAT "linux") else() # We don't do distribution for Android # Set for completeness set(JAVA_PLAT "android") endif() # Similar to Nuget schema set(JAVA_OS_ARCH ${JAVA_PLAT}-${JNI_ARCH}) # expose native libraries to the gradle build process set(JAVA_PACKAGE_DIR ai/onnxruntime/extensions/native/${JAVA_OS_ARCH}) set(JAVA_NATIVE_LIB_DIR ${JAVA_OUTPUT_DIR}/native-lib) set(JAVA_NATIVE_JNI_DIR ${JAVA_OUTPUT_DIR}/native-jni) set(JAVA_PACKAGE_LIB_DIR ${JAVA_NATIVE_LIB_DIR}/${JAVA_PACKAGE_DIR}) set(JAVA_PACKAGE_JNI_DIR ${JAVA_NATIVE_JNI_DIR}/${JAVA_PACKAGE_DIR}) file(MAKE_DIRECTORY ${JAVA_PACKAGE_LIB_DIR}) file(MAKE_DIRECTORY ${JAVA_PACKAGE_JNI_DIR}) # On Windows TARGET_LINKER_FILE_NAME is the .lib, TARGET_FILE_NAME is the .dll if (WIN32) #Our static analysis plugin set /p:LinkCompiled=false if(NOT onnxruntime_extensions_ENABLE_STATIC_ANALYSIS) add_custom_command(TARGET onnxruntime_extensions4j_jni POST_BUILD COMMAND ${CMAKE_COMMAND} -E copy_if_different $ ${JAVA_PACKAGE_JNI_DIR}/$) endif() else() add_custom_command(TARGET onnxruntime_extensions4j_jni POST_BUILD COMMAND ${CMAKE_COMMAND} -E copy_if_different $ ${JAVA_PACKAGE_JNI_DIR}/$) endif() # run the build process (this copies the results back into CMAKE_CURRENT_BINARY_DIR) set(GRADLE_ARGS --console=plain cmakeBuild -p ${JAVA_ROOT} -DcmakeBuildDir=${CMAKE_CURRENT_BINARY_DIR}) if(WIN32) set(GRADLE_ARGS ${GRADLE_ARGS} -Dorg.gradle.daemon=false) elseif (ANDROID) # For Android build, we may run gradle multiple times in same build, # sometimes gradle JVM will run out of memory if we keep the daemon running # it is better to not keep a daemon running set(GRADLE_ARGS ${GRADLE_ARGS} --no-daemon) endif() message(STATUS "GRADLE_ARGS: ${GRADLE_ARGS}") add_custom_command(TARGET onnxruntime_extensions4j_jni POST_BUILD COMMAND ${GRADLE_EXECUTABLE} ${GRADLE_ARGS} WORKING_DIRECTORY ${JAVA_OUTPUT_TEMP}) if (ANDROID) set(ANDROID_PACKAGE_JNILIBS_DIR ${JAVA_OUTPUT_DIR}/android) set(ANDROID_PACKAGE_ABI_DIR ${ANDROID_PACKAGE_JNILIBS_DIR}/${ANDROID_ABI}) # Copy onnxruntime_extensions4j_jni.so and ortextensions.so for building Android AAR package and use in NuGet add_custom_command(TARGET onnxruntime_extensions4j_jni POST_BUILD COMMAND ${CMAKE_COMMAND} -E make_directory ${ANDROID_PACKAGE_ABI_DIR} COMMAND ${CMAKE_COMMAND} -E copy_if_different $ ${ANDROID_PACKAGE_ABI_DIR}/$) if (_BUILD_SHARED_LIBRARY) add_custom_command(TARGET onnxruntime_extensions4j_jni POST_BUILD COMMAND ${CMAKE_COMMAND} -E copy_if_different $ ${ANDROID_PACKAGE_ABI_DIR}/$) endif() if (OCOS_ENABLE_AZURE) add_custom_command(TARGET onnxruntime_extensions4j_jni POST_BUILD COMMAND ${CMAKE_COMMAND} -E copy_if_different $ ${ANDROID_PACKAGE_ABI_DIR}/$ COMMAND ${CMAKE_COMMAND} -E copy_if_different $ ${ANDROID_PACKAGE_ABI_DIR}/$ # not sure why but we need to use the library name directly for curl instead of CURL::libcurl COMMAND ${CMAKE_COMMAND} -E copy_if_different ${CURL_ROOT_DIR}/lib/libcurl.so ${ANDROID_PACKAGE_ABI_DIR}/libcurl.so ) endif() endif()