123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182 |
- # Distributed under the OSI-approved MIT License. See accompanying
- # file LICENSE or https://github.com/Crascit/DownloadProject for details.
- #
- # MODULE: DownloadProject
- #
- # PROVIDES:
- # download_project( PROJ projectName
- # [PREFIX prefixDir]
- # [DOWNLOAD_DIR downloadDir]
- # [SOURCE_DIR srcDir]
- # [BINARY_DIR binDir]
- # [QUIET]
- # ...
- # )
- #
- # Provides the ability to download and unpack a tarball, zip file, git repository,
- # etc. at configure time (i.e. when the cmake command is run). How the downloaded
- # and unpacked contents are used is up to the caller, but the motivating case is
- # to download source code which can then be included directly in the build with
- # add_subdirectory() after the call to download_project(). Source and build
- # directories are set up with this in mind.
- #
- # The PROJ argument is required. The projectName value will be used to construct
- # the following variables upon exit (obviously replace projectName with its actual
- # value):
- #
- # projectName_SOURCE_DIR
- # projectName_BINARY_DIR
- #
- # The SOURCE_DIR and BINARY_DIR arguments are optional and would not typically
- # need to be provided. They can be specified if you want the downloaded source
- # and build directories to be located in a specific place. The contents of
- # projectName_SOURCE_DIR and projectName_BINARY_DIR will be populated with the
- # locations used whether you provide SOURCE_DIR/BINARY_DIR or not.
- #
- # The DOWNLOAD_DIR argument does not normally need to be set. It controls the
- # location of the temporary CMake build used to perform the download.
- #
- # The PREFIX argument can be provided to change the base location of the default
- # values of DOWNLOAD_DIR, SOURCE_DIR and BINARY_DIR. If all of those three arguments
- # are provided, then PREFIX will have no effect. The default value for PREFIX is
- # CMAKE_BINARY_DIR.
- #
- # The QUIET option can be given if you do not want to show the output associated
- # with downloading the specified project.
- #
- # In addition to the above, any other options are passed through unmodified to
- # ExternalProject_Add() to perform the actual download, patch and update steps.
- # The following ExternalProject_Add() options are explicitly prohibited (they
- # are reserved for use by the download_project() command):
- #
- # CONFIGURE_COMMAND
- # BUILD_COMMAND
- # INSTALL_COMMAND
- # TEST_COMMAND
- #
- # Only those ExternalProject_Add() arguments which relate to downloading, patching
- # and updating of the project sources are intended to be used. Also note that at
- # least one set of download-related arguments are required.
- #
- # If using CMake 3.2 or later, the UPDATE_DISCONNECTED option can be used to
- # prevent a check at the remote end for changes every time CMake is run
- # after the first successful download. See the documentation of the ExternalProject
- # module for more information. It is likely you will want to use this option if it
- # is available to you. Note, however, that the ExternalProject implementation contains
- # bugs which result in incorrect handling of the UPDATE_DISCONNECTED option when
- # using the URL download method or when specifying a SOURCE_DIR with no download
- # method. Fixes for these have been created, the last of which is scheduled for
- # inclusion in CMake 3.8.0. Details can be found here:
- #
- # https://gitlab.kitware.com/cmake/cmake/commit/bdca68388bd57f8302d3c1d83d691034b7ffa70c
- # https://gitlab.kitware.com/cmake/cmake/issues/16428
- #
- # If you experience build errors related to the update step, consider avoiding
- # the use of UPDATE_DISCONNECTED.
- #
- # EXAMPLE USAGE:
- #
- # include(DownloadProject)
- # download_project(PROJ googletest
- # GIT_REPOSITORY https://github.com/google/googletest.git
- # GIT_TAG master
- # UPDATE_DISCONNECTED 1
- # QUIET
- # )
- #
- # add_subdirectory(${googletest_SOURCE_DIR} ${googletest_BINARY_DIR})
- #
- #========================================================================================
- set(_DownloadProjectDir "${CMAKE_CURRENT_LIST_DIR}")
- include(CMakeParseArguments)
- function(download_project)
- set(options QUIET)
- set(oneValueArgs
- PROJ
- PREFIX
- DOWNLOAD_DIR
- SOURCE_DIR
- BINARY_DIR
- # Prevent the following from being passed through
- CONFIGURE_COMMAND
- BUILD_COMMAND
- INSTALL_COMMAND
- TEST_COMMAND
- )
- set(multiValueArgs "")
- cmake_parse_arguments(DL_ARGS "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
- # Hide output if requested
- if (DL_ARGS_QUIET)
- set(OUTPUT_QUIET "OUTPUT_QUIET")
- else()
- unset(OUTPUT_QUIET)
- message(STATUS "Downloading/updating ${DL_ARGS_PROJ}")
- endif()
- # Set up where we will put our temporary CMakeLists.txt file and also
- # the base point below which the default source and binary dirs will be.
- # The prefix must always be an absolute path.
- if (NOT DL_ARGS_PREFIX)
- set(DL_ARGS_PREFIX "${CMAKE_BINARY_DIR}")
- else()
- get_filename_component(DL_ARGS_PREFIX "${DL_ARGS_PREFIX}" ABSOLUTE
- BASE_DIR "${CMAKE_CURRENT_BINARY_DIR}")
- endif()
- if (NOT DL_ARGS_DOWNLOAD_DIR)
- set(DL_ARGS_DOWNLOAD_DIR "${DL_ARGS_PREFIX}/${DL_ARGS_PROJ}-download")
- endif()
- # Ensure the caller can know where to find the source and build directories
- if (NOT DL_ARGS_SOURCE_DIR)
- set(DL_ARGS_SOURCE_DIR "${DL_ARGS_PREFIX}/${DL_ARGS_PROJ}-src")
- endif()
- if (NOT DL_ARGS_BINARY_DIR)
- set(DL_ARGS_BINARY_DIR "${DL_ARGS_PREFIX}/${DL_ARGS_PROJ}-build")
- endif()
- set(${DL_ARGS_PROJ}_SOURCE_DIR "${DL_ARGS_SOURCE_DIR}" PARENT_SCOPE)
- set(${DL_ARGS_PROJ}_BINARY_DIR "${DL_ARGS_BINARY_DIR}" PARENT_SCOPE)
- # The way that CLion manages multiple configurations, it causes a copy of
- # the CMakeCache.txt to be copied across due to it not expecting there to
- # be a project within a project. This causes the hard-coded paths in the
- # cache to be copied and builds to fail. To mitigate this, we simply
- # remove the cache if it exists before we configure the new project. It
- # is safe to do so because it will be re-generated. Since this is only
- # executed at the configure step, it should not cause additional builds or
- # downloads.
- file(REMOVE "${DL_ARGS_DOWNLOAD_DIR}/CMakeCache.txt")
- # Create and build a separate CMake project to carry out the download.
- # If we've already previously done these steps, they will not cause
- # anything to be updated, so extra rebuilds of the project won't occur.
- # Make sure to pass through CMAKE_MAKE_PROGRAM in case the main project
- # has this set to something not findable on the PATH.
- configure_file("${_DownloadProjectDir}/DownloadProject.CMakeLists.cmake.in"
- "${DL_ARGS_DOWNLOAD_DIR}/CMakeLists.txt")
- execute_process(COMMAND ${CMAKE_COMMAND} -G "${CMAKE_GENERATOR}"
- -D "CMAKE_MAKE_PROGRAM:FILE=${CMAKE_MAKE_PROGRAM}"
- .
- RESULT_VARIABLE result
- ${OUTPUT_QUIET}
- WORKING_DIRECTORY "${DL_ARGS_DOWNLOAD_DIR}"
- )
- if(result)
- message(FATAL_ERROR "CMake step for ${DL_ARGS_PROJ} failed: ${result}")
- endif()
- execute_process(COMMAND ${CMAKE_COMMAND} --build .
- RESULT_VARIABLE result
- ${OUTPUT_QUIET}
- WORKING_DIRECTORY "${DL_ARGS_DOWNLOAD_DIR}"
- )
- if(result)
- message(FATAL_ERROR "Build step for ${DL_ARGS_PROJ} failed: ${result}")
- endif()
- endfunction()
|