3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-04-06 17:44:08 +00:00

[CMake] Teach CMake to support git worktrees. This fixes the bug

reported by @nbraud reported in #1227.

Previously the CMake build system assumed that the `.git` file must
be a directory. This is not the case when the working directory
is a "git worktree". In this case the `.git` file is just a plain
file that points to a directory within the true `.git` directory.
This commit essentially implements the logic to traverse this extra
level of indirection and removes some assumptions that the `.git`
file is a directory.
This commit is contained in:
Dan Liew 2017-08-23 19:21:32 +01:00
parent ee00852151
commit 58f152a92a

View file

@ -4,23 +4,55 @@
# of the git directory changes CMake will be forced to re-run. This useful # of the git directory changes CMake will be forced to re-run. This useful
# for fetching the current git hash and including it in the build. # for fetching the current git hash and including it in the build.
# #
# `GIT_DIR` is the path to the git directory (i.e. the `.git` directory) # `GIT_DOT_FILE` is the path to the git directory (i.e. the `.git` directory) or
# `.git` file used by a git worktree.
# `SUCCESS_VAR` is the name of the variable to set. It will be set to TRUE # `SUCCESS_VAR` is the name of the variable to set. It will be set to TRUE
# if the dependency was successfully added and FALSE otherwise. # if the dependency was successfully added and FALSE otherwise.
function(add_git_dir_dependency GIT_DIR SUCCESS_VAR) function(add_git_dir_dependency GIT_DOT_FILE SUCCESS_VAR)
if (NOT "${ARGC}" EQUAL 2) if (NOT "${ARGC}" EQUAL 2)
message(FATAL_ERROR "Invalid number (${ARGC}) of arguments") message(FATAL_ERROR "Invalid number (${ARGC}) of arguments")
endif() endif()
if (NOT IS_ABSOLUTE "${GIT_DIR}") if (NOT IS_ABSOLUTE "${GIT_DOT_FILE}")
message(FATAL_ERROR "GIT_DIR (\"${GIT_DIR}\") is not an absolute path") message(FATAL_ERROR "GIT_DOT_FILE (\"${GIT_DOT_FILE}\") is not an absolute path")
endif() endif()
if (NOT IS_DIRECTORY "${GIT_DIR}") if (NOT EXISTS "${GIT_DOT_FILE}")
message(FATAL_ERROR "GIT_DIR (\"${GIT_DIR}\") is not a directory") message(FATAL_ERROR "GIT_DOT_FILE (\"${GIT_DOT_FILE}\") does not exist")
endif() endif()
set(GIT_HEAD_FILE "${GIT_DIR}/HEAD") if (NOT IS_DIRECTORY "${GIT_DOT_FILE}")
# Might be a git worktree. In this case we need parse out the worktree
# git directory
file(READ "${GIT_DOT_FILE}" GIT_DOT_FILE_DATA LIMIT 512)
string(STRIP "${GIT_DOT_FILE_DATA}" GIT_DOT_FILE_DATA_STRIPPED)
if ("${GIT_DOT_FILE_DATA_STRIPPED}" MATCHES "^gitdir:[ ]*(.+)$")
# Git worktree
message(STATUS "Found git worktree")
set(GIT_WORKTREE_DIR "${CMAKE_MATCH_1}")
set(GIT_HEAD_FILE "${GIT_WORKTREE_DIR}/HEAD")
# Figure out where real git directory lives
set(GIT_COMMON_DIR_FILE "${GIT_WORKTREE_DIR}/commondir")
if (NOT EXISTS "${GIT_COMMON_DIR_FILE}")
message(FATAL_ERROR "Found git worktree dir but could not find \"${GIT_COMMON_DIR_FILE}\"")
endif()
file(READ "${GIT_COMMON_DIR_FILE}" GIT_COMMON_DIR_FILE_DATA LIMIT 512)
string(STRIP "${GIT_COMMON_DIR_FILE_DATA}" GIT_COMMON_DIR_FILE_DATA_STRIPPED)
get_filename_component(GIT_DIR "${GIT_WORKTREE_DIR}/${GIT_COMMON_DIR_FILE_DATA_STRIPPED}" ABSOLUTE)
if (NOT IS_DIRECTORY "${GIT_DIR}")
message(FATAL_ERROR "Failed to compute path to git directory from git worktree")
endif()
else()
message(FATAL_ERROR "GIT_DOT_FILE (\"${GIT_DOT_FILE}\") is not a directory or a pointer to git worktree directory")
endif()
else()
# Just a normal `.git` directory
message(STATUS "Found simple git working directory")
set(GIT_HEAD_FILE "${GIT_DOT_FILE}/HEAD")
set(GIT_DIR "${GIT_DOT_FILE}")
endif()
message(STATUS "Found git directory \"${GIT_DIR}\"")
if (NOT EXISTS "${GIT_HEAD_FILE}") if (NOT EXISTS "${GIT_HEAD_FILE}")
message(AUTHOR_WARNING "Git head file \"${GIT_HEAD_FILE}\" cannot be found") message(AUTHOR_WARNING "Git head file \"${GIT_HEAD_FILE}\" cannot be found")
set(${SUCCESS_VAR} FALSE PARENT_SCOPE) set(${SUCCESS_VAR} FALSE PARENT_SCOPE)
@ -79,24 +111,25 @@ function(add_git_dir_dependency GIT_DIR SUCCESS_VAR)
set(${SUCCESS_VAR} TRUE PARENT_SCOPE) set(${SUCCESS_VAR} TRUE PARENT_SCOPE)
endfunction() endfunction()
# get_git_head_hash(GIT_DIR OUTPUT_VAR) # get_git_head_hash(GIT_DOT_FILE OUTPUT_VAR)
# #
# Retrieve the current commit hash for a git working directory where `GIT_DIR` # Retrieve the current commit hash for a git working directory where
# is the `.git` directory in the root of the git working directory. # `GIT_DOT_FILE` is the `.git` directory or `.git` pointer file in a git
# worktree in the root of the git working directory.
# #
# `OUTPUT_VAR` should be the name of the variable to put the result in. If this # `OUTPUT_VAR` should be the name of the variable to put the result in. If this
# function fails then either a fatal error will be raised or `OUTPUT_VAR` will # function fails then either a fatal error will be raised or `OUTPUT_VAR` will
# contain a string with the suffix `NOTFOUND` which can be used in CMake `if()` # contain a string with the suffix `NOTFOUND` which can be used in CMake `if()`
# commands. # commands.
function(get_git_head_hash GIT_DIR OUTPUT_VAR) function(get_git_head_hash GIT_DOT_FILE OUTPUT_VAR)
if (NOT "${ARGC}" EQUAL 2) if (NOT "${ARGC}" EQUAL 2)
message(FATAL_ERROR "Invalid number of arguments") message(FATAL_ERROR "Invalid number of arguments")
endif() endif()
if (NOT IS_DIRECTORY "${GIT_DIR}") if (NOT EXISTS "${GIT_DOT_FILE}")
message(FATAL_ERROR "\"${GIT_DIR}\" is not a directory") message(FATAL_ERROR "\"${GIT_DOT_FILE}\" does not exist")
endif() endif()
if (NOT IS_ABSOLUTE "${GIT_DIR}") if (NOT IS_ABSOLUTE "${GIT_DOT_FILE}")
message(FATAL_ERROR \""${GIT_DIR}\" is not an absolute path") message(FATAL_ERROR \""${GIT_DOT_FILE}\" is not an absolute path")
endif() endif()
find_package(Git) find_package(Git)
# NOTE: Use `GIT_FOUND` rather than `Git_FOUND` which was only # NOTE: Use `GIT_FOUND` rather than `Git_FOUND` which was only
@ -105,7 +138,7 @@ function(get_git_head_hash GIT_DIR OUTPUT_VAR)
set(${OUTPUT_VAR} "GIT-NOTFOUND" PARENT_SCOPE) set(${OUTPUT_VAR} "GIT-NOTFOUND" PARENT_SCOPE)
return() return()
endif() endif()
get_filename_component(GIT_WORKING_DIR "${GIT_DIR}" DIRECTORY) get_filename_component(GIT_WORKING_DIR "${GIT_DOT_FILE}" DIRECTORY)
execute_process( execute_process(
COMMAND COMMAND
"${GIT_EXECUTABLE}" "${GIT_EXECUTABLE}"
@ -128,24 +161,25 @@ function(get_git_head_hash GIT_DIR OUTPUT_VAR)
set(${OUTPUT_VAR} "${Z3_GIT_HASH}" PARENT_SCOPE) set(${OUTPUT_VAR} "${Z3_GIT_HASH}" PARENT_SCOPE)
endfunction() endfunction()
# get_git_head_describe(GIT_DIR OUTPUT_VAR) # get_git_head_describe(GIT_DOT_FILE OUTPUT_VAR)
# #
# Retrieve the output of `git describe` for a git working directory where # Retrieve the output of `git describe` for a git working directory where
# `GIT_DIR` is the `.git` directory in the root of the git working directory. # `GIT_DOT_FILE` is the `.git` directory or `.git` pointer file in a git
# worktree in the root of the git working directory.
# #
# `OUTPUT_VAR` should be the name of the variable to put the result in. If this # `OUTPUT_VAR` should be the name of the variable to put the result in. If this
# function fails then either a fatal error will be raised or `OUTPUT_VAR` will # function fails then either a fatal error will be raised or `OUTPUT_VAR` will
# contain a string with the suffix `NOTFOUND` which can be used in CMake `if()` # contain a string with the suffix `NOTFOUND` which can be used in CMake `if()`
# commands. # commands.
function(get_git_head_describe GIT_DIR OUTPUT_VAR) function(get_git_head_describe GIT_DOT_FILE OUTPUT_VAR)
if (NOT "${ARGC}" EQUAL 2) if (NOT "${ARGC}" EQUAL 2)
message(FATAL_ERROR "Invalid number of arguments") message(FATAL_ERROR "Invalid number of arguments")
endif() endif()
if (NOT IS_DIRECTORY "${GIT_DIR}") if (NOT EXISTS "${GIT_DOT_FILE}")
message(FATAL_ERROR "\"${GIT_DIR}\" is not a directory") message(FATAL_ERROR "\"${GIT_DOT_FILE}\" does not exist")
endif() endif()
if (NOT IS_ABSOLUTE "${GIT_DIR}") if (NOT IS_ABSOLUTE "${GIT_DOT_FILE}")
message(FATAL_ERROR \""${GIT_DIR}\" is not an absolute path") message(FATAL_ERROR \""${GIT_DOT_FILE}\" is not an absolute path")
endif() endif()
find_package(Git) find_package(Git)
# NOTE: Use `GIT_FOUND` rather than `Git_FOUND` which was only # NOTE: Use `GIT_FOUND` rather than `Git_FOUND` which was only
@ -154,7 +188,7 @@ function(get_git_head_describe GIT_DIR OUTPUT_VAR)
set(${OUTPUT_VAR} "GIT-NOTFOUND" PARENT_SCOPE) set(${OUTPUT_VAR} "GIT-NOTFOUND" PARENT_SCOPE)
return() return()
endif() endif()
get_filename_component(GIT_WORKING_DIR "${GIT_DIR}" DIRECTORY) get_filename_component(GIT_WORKING_DIR "${GIT_DOT_FILE}" DIRECTORY)
execute_process( execute_process(
COMMAND COMMAND
"${GIT_EXECUTABLE}" "${GIT_EXECUTABLE}"