|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
# CMake file for Matlab/Octave support
|
|
|
|
#
|
|
|
|
# Matlab code generation and compilation is broken down into two distinct
|
|
|
|
# stages: configure time and build time. The idea is that we want to give
|
|
|
|
# the user reasonable guarantees that once they type 'make', wrapper
|
|
|
|
# generation is unlikely to fail. Therefore we run a series of tests at
|
|
|
|
# configure time to check the working status of the core components.
|
|
|
|
#
|
|
|
|
# Configure Time
|
|
|
|
# During configure time, the script attempts to ascertain whether the
|
|
|
|
# generator and mex compiler are working for a given architecture.
|
|
|
|
# Currently this involves:
|
|
|
|
# 1) Generating a simple CV_EXPORTS_W symbol and checking whether a file
|
|
|
|
# of the symbol name is generated
|
|
|
|
# 2) Compiling a simple mex gateway to check that Bridge.hpp and mex.h
|
|
|
|
# can be found, and that a file with the mexext is produced
|
|
|
|
#
|
|
|
|
# Build Time
|
|
|
|
# If the configure time tests pass, then we assume Matlab wrapper generation
|
|
|
|
# will not fail during build time. We simply glob all of the symbols in
|
|
|
|
# the OpenCV module headers, generate intermediate .cpp files, then compile
|
|
|
|
# them with mex.
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
# PREPEND
|
|
|
|
# Given a list of strings IN and a TOKEN, prepend the token to each string
|
|
|
|
# and append to OUT. This is used for passing command line "-I", "-L" and "-l"
|
|
|
|
# arguments to mex. e.g.
|
|
|
|
# prepend("-I" OUT /path/to/include/dir) --> -I/path/to/include/dir
|
|
|
|
macro(PREPEND TOKEN OUT IN)
|
|
|
|
foreach(VAR ${IN})
|
|
|
|
list(APPEND ${OUT} "${TOKEN}${VAR}")
|
|
|
|
endforeach()
|
|
|
|
endmacro()
|
|
|
|
|
|
|
|
# WARN_MIXED_PRECISION
|
|
|
|
# Formats a warning message if the compiler and Matlab bitness is different
|
|
|
|
macro(WARN_MIXED_PRECISION COMPILER_BITNESS MATLAB_BITNESS)
|
|
|
|
set(MSG "Your compiler is ${COMPILER_BITNESS}-bit")
|
|
|
|
set(MSG "${MSG} but your version of Matlab is ${MATLAB_BITNESS}-bit.")
|
|
|
|
set(MSG "${MSG} Mixed preicision pointers are not supported. Disabling Matlab bindings...")
|
|
|
|
message(WARNING ${MSG})
|
|
|
|
endmacro()
|
|
|
|
|
|
|
|
# make sure we're on a supported architecture with Matlab and python installed
|
|
|
|
if (IOS OR ANDROID OR NOT MATLAB_FOUND OR NOT PYTHONLIBS_FOUND)
|
|
|
|
ocv_module_disable(matlab)
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
# If the user built OpenCV as X-bit, but they have a Y-bit version of Matlab,
|
|
|
|
# attempting to link to OpenCV during binding generation will fail, since
|
|
|
|
# mixed precision pointers are not allowed. Disable the bindings.
|
|
|
|
if (${CMAKE_SIZEOF_VOID_P} EQUAL 4 AND ${MATLAB_ARCH} MATCHES "64")
|
|
|
|
warn_mixed_precision("32" "64")
|
|
|
|
ocv_module_disable(matlab)
|
|
|
|
return()
|
|
|
|
elseif (${CMAKE_SIZEOF_VOID_P} EQUAL 8 AND NOT ${MATLAB_ARCH} MATCHES "64")
|
|
|
|
warn_mixed_precision("64" "32")
|
|
|
|
ocv_module_disable(matlab)
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
set(the_description "The Matlab/Octave bindings")
|
|
|
|
ocv_add_module(matlab BINDINGS
|
|
|
|
OPTIONAL opencv_core
|
|
|
|
opencv_imgproc opencv_ml opencv_highgui
|
|
|
|
opencv_objdetect opencv_flann opencv_features2d
|
|
|
|
opencv_photo opencv_video opencv_videostab
|
|
|
|
opencv_calib opencv_calib3d
|
|
|
|
opencv_stitching opencv_superres
|
|
|
|
opencv_nonfree
|
|
|
|
)
|
|
|
|
|
|
|
|
# set the path to the C++ header and doc parser
|
|
|
|
set(HDR_PARSER_PATH ${CMAKE_SOURCE_DIR}/modules/python/src2)
|
|
|
|
set(RST_PARSER_PATH ${CMAKE_SOURCE_DIR}/modules/java/generator)
|
|
|
|
|
|
|
|
# set mex compiler options
|
|
|
|
prepend("-I" MEX_INCLUDE_DIRS ${CMAKE_CURRENT_SOURCE_DIR}/include)
|
|
|
|
prepend("-L" MEX_LIB_DIR ${LIBRARY_OUTPUT_PATH}/$(Configuration))
|
|
|
|
set(MEX_OPTS -largeArrayDims)
|
|
|
|
|
|
|
|
if (BUILD_TESTS)
|
|
|
|
add_subdirectory(test)
|
|
|
|
endif()
|
|
|
|
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/include)
|
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
# Configure time components
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
set(MATLAB_DEPS ${OPENCV_MODULE_${the_module}_REQ_DEPS} ${OPENCV_MODULE_${the_module}_OPT_DEPS})
|
|
|
|
foreach(opencv_module ${MATLAB_DEPS})
|
|
|
|
if (HAVE_${opencv_module})
|
|
|
|
string(REPLACE "opencv_" "" module ${opencv_module})
|
|
|
|
list(APPEND opencv_modules ${module})
|
|
|
|
list(APPEND ${the_module}_ACTUAL_DEPS ${opencv_module})
|
|
|
|
prepend("-I" MEX_INCLUDE_DIRS "${OPENCV_MODULE_${opencv_module}_LOCATION}/include")
|
|
|
|
prepend("-l" MEX_LIBS ${opencv_module}${OPENCV_DLLVERSION})
|
|
|
|
prepend("-l" MEX_DEBUG_LIBS ${opencv_module}${OPENCV_DLLVERSION}${OPENCV_DEBUG_POSTFIX})
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
|
|
|
|
# add extra headers by hand
|
|
|
|
list(APPEND opencv_extra_hdrs "core:${OPENCV_MODULE_opencv_core_LOCATION}/include/opencv2/core/base.hpp")
|
|
|
|
|
|
|
|
# pass the OPENCV_CXX_EXTRA_FLAGS through to the mex compiler
|
|
|
|
# remove the visibility modifiers, so the mex gateway is visible
|
|
|
|
string(REGEX REPLACE "[^\ ]*visibility[^\ ]*" "" MEX_CXX_FLAGS "${OPENCV_EXTRA_FLAGS} ${OPENCV_EXTRA_CXX_FLAGS}")
|
|
|
|
|
|
|
|
# Configure checks
|
|
|
|
# Check to see whether the generator and the mex compiler are working.
|
|
|
|
# The checks currently test:
|
|
|
|
# - whether the python generator can be found
|
|
|
|
# - whether the python generator correctly outputs a file for a definition
|
|
|
|
# - whether the mex compiler can find the required headers
|
|
|
|
# - whether the mex compiler can compile a trivial definition
|
|
|
|
if (NOT MEX_WORKS)
|
|
|
|
# attempt to generate a gateway for a function
|
|
|
|
message(STATUS "Trying to generate Matlab code")
|
|
|
|
execute_process(
|
|
|
|
COMMAND ${PYTHON_EXECUTABLE}
|
|
|
|
${CMAKE_CURRENT_SOURCE_DIR}/generator/gen_matlab.py
|
|
|
|
--hdrparser ${HDR_PARSER_PATH}
|
|
|
|
--rstparser ${RST_PARSER_PATH}
|
|
|
|
--extra "test:${CMAKE_CURRENT_SOURCE_DIR}/test/test_generator.hpp"
|
|
|
|
--outdir ${CMAKE_BINARY_DIR}/junk
|
|
|
|
ERROR_VARIABLE GEN_ERROR
|
|
|
|
OUTPUT_QUIET
|
|
|
|
)
|
|
|
|
|
|
|
|
if (GEN_ERROR)
|
|
|
|
message(${GEN_ERROR})
|
|
|
|
message(STATUS "Error generating Matlab code. Disabling Matlab bindings...")
|
|
|
|
ocv_module_disable(matlab)
|
|
|
|
return()
|
|
|
|
else()
|
|
|
|
message(STATUS "Trying to generate Matlab code - OK")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
# attempt to compile a gateway using mex
|
|
|
|
message(STATUS "Trying to compile mex file")
|
|
|
|
execute_process(
|
|
|
|
COMMAND ${MATLAB_MEX_SCRIPT} ${MEX_OPTS} "CXXFLAGS=\$CXXFLAGS ${MEX_CXXFLAGS}"
|
|
|
|
${MEX_INCLUDE_DIRS} ${CMAKE_CURRENT_SOURCE_DIR}/test/test_compiler.cpp
|
|
|
|
WORKING_DIRECTORY ${CMAKE_BINARY_DIR}/junk
|
|
|
|
ERROR_VARIABLE MEX_ERROR
|
|
|
|
OUTPUT_QUIET
|
|
|
|
)
|
|
|
|
|
|
|
|
if (MEX_ERROR)
|
|
|
|
message(${MEX_ERROR})
|
|
|
|
message(STATUS "Error compiling mex file. Disabling Matlab bindings...")
|
|
|
|
ocv_module_disable(matlab)
|
|
|
|
return()
|
|
|
|
else()
|
|
|
|
message(STATUS "Trying to compile mex file - OK")
|
|
|
|
endif()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
# if we make it here, mex works!
|
|
|
|
set_property(GLOBAL PROPERTY MEX_WORKS TRUE)
|
|
|
|
set(MEX_WORKS True CACHE BOOL ADVANCED)
|
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
# Build time components
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
# proxies
|
|
|
|
# these proxies are used to trigger the add_custom_commands
|
|
|
|
# (which do the real work) only when they're outdated
|
|
|
|
set(GENERATE_PROXY ${CMAKE_CURRENT_BINARY_DIR}/generate.proxy)
|
|
|
|
set(COMPILE_PROXY ${CMAKE_CURRENT_BINARY_DIR}/compile.proxy)
|
|
|
|
# TODO: Remove following line before merging with master
|
|
|
|
file(REMOVE ${GENERATE_PROXY} ${COMPILE_PROXY})
|
|
|
|
|
|
|
|
# generate
|
|
|
|
# call the python executable to generate the Matlab gateways
|
|
|
|
add_custom_command(
|
|
|
|
OUTPUT ${GENERATE_PROXY}
|
|
|
|
COMMAND ${PYTHON_EXECUTABLE}
|
|
|
|
${CMAKE_CURRENT_SOURCE_DIR}/generator/gen_matlab.py
|
|
|
|
--hdrparser ${HDR_PARSER_PATH}
|
|
|
|
--rstparser ${RST_PARSER_PATH}
|
|
|
|
--moduleroot ${CMAKE_SOURCE_DIR}/modules
|
|
|
|
--modules ${opencv_modules}
|
|
|
|
--extra ${opencv_extra_hdrs}
|
|
|
|
--outdir ${CMAKE_CURRENT_BINARY_DIR}
|
|
|
|
COMMAND ${CMAKE_COMMAND} -E touch ${GENERATE_PROXY}
|
|
|
|
COMMENT "Generating Matlab source files"
|
|
|
|
)
|
|
|
|
|
|
|
|
# compile
|
|
|
|
# call the mex compiler to compile the gateways
|
|
|
|
# because we don't know the source files at configure-time, this
|
|
|
|
# has to be executed in a separate script in cmake's script processing mode
|
|
|
|
add_custom_command(
|
|
|
|
OUTPUT ${COMPILE_PROXY}
|
|
|
|
COMMAND ${CMAKE_COMMAND} -DMATLAB_MEX_SCRIPT=${MATLAB_MEX_SCRIPT}
|
|
|
|
-DMATLAB_MEXEXT=${MATLAB_MEXEXT}
|
|
|
|
-DMEX_OPTS=${MEX_OPTS}
|
|
|
|
-DMEX_CXXFLAGS=${MEX_CXXFLAGS}
|
|
|
|
-DMEX_INCLUDE_DIRS="${MEX_INCLUDE_DIRS}"
|
|
|
|
-DMEX_LIB_DIR=${MEX_LIB_DIR}
|
|
|
|
-DCONFIGURATION="$(Configuration)"
|
|
|
|
-DMEX_LIBS="${MEX_LIBS}"
|
|
|
|
-DMEX_DEBUG_LIBS="${MEX_DEBUG_LIBS}"
|
|
|
|
-P ${CMAKE_CURRENT_SOURCE_DIR}/compile.cmake
|
|
|
|
COMMAND ${CMAKE_COMMAND} -E touch ${COMPILE_PROXY}
|
|
|
|
COMMENT "Compiling Matlab source files. This could take a while..."
|
|
|
|
)
|
|
|
|
|
|
|
|
# targets
|
|
|
|
# opencv_matlab_sources --> opencv_matlab
|
|
|
|
add_custom_target(${the_module}_sources ALL DEPENDS ${GENERATE_PROXY})
|
|
|
|
add_custom_target(${the_module} ALL DEPENDS ${COMPILE_PROXY})
|
|
|
|
add_dependencies(${the_module} ${the_module}_sources ${${the_module}_ACTUAL_DEPS})
|
|
|
|
|
|
|
|
if (ENABLE_SOLUTION_FOLDERS)
|
|
|
|
set_target_properties(${the_module} PROPERTIES FOLDER "modules")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
# Install time components
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
# NOTE: Trailing slashes on the DIRECTORY paths are important!
|
|
|
|
# TODO: What needs to be done with rpath????
|
|
|
|
file(GLOB MATLAB_FUNCTIONS "${CMAKE_CURRENT_BINARY_DIR}/src/*.${MATLAB_MEXEXT}")
|
|
|
|
file(GLOB MATLAB_CLASSES "${CMAKE_CURRENT_BINARY_DIR}/+cv/*.m")
|
|
|
|
file(GLOB MATLAB_PRIVATE "${CMAKE_CURRENT_BINARY_DIR}/src/private/*.${MATLAB_MEXEXT}")
|
|
|
|
install(DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/+cv/
|
|
|
|
DESTINATION ${CMAKE_INSTALL_PREFIX}/matlab/+cv
|
|
|
|
FILES_MATCHING PATTERN "*.${MATLAB_MEXEXT}"
|
|
|
|
)
|
|
|
|
install(DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/+cv/
|
|
|
|
DESTINATION ${CMAKE_INSTALL_PREFIX}/matlab/+cv
|
|
|
|
FILES_MATCHING PATTERN "*.m"
|
|
|
|
)
|
|
|
|
install(DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/+cv/private/
|
|
|
|
DESTINATION ${CMAKE_INSTALL_PREFIX}/matlab/+cv/private
|
|
|
|
FILES_MATCHING PATTERN "*.${MATLAB_MEXEXT}"
|
|
|
|
)
|
|
|
|
install(DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/
|
|
|
|
DESTINATION ${CMAKE_INSTALL_PREFIX}/matlab
|
|
|
|
FILES_MATCHING PATTERN "cv.m"
|
|
|
|
)
|