sgraenitz created this revision.
sgraenitz added reviewers: xiaobai, compnerd, JDevlieghere, labath.
Herald added subscribers: kristof.beyls, javed.absar, mgorny, ki.stfu.
Herald added a project: LLDB.

We got used to post-build commands for copying resources into the build-tree 
framework. Executables were included by adding their target names to the list 
of `LLDB_FRAMEWORK_TOOLS`. Install destinations were set in 
`add_lldb_tool(<target-name> ...)`.

This patch unifies these steps in `lldb_add_to_framework()` which:

- adds a copy to the build-tree framework for testing
- adds a cleanup step to run before install
- sets the target's destination in the install-tree

Key changes:

- `LLDB_BUILD_FRAMEWORK` now disables the default `GENERATE_INSTALL` logic
- `LLDB_FRAMEWORK_TOOLS` is obsolete; instead each tool calls 
`lldb_add_to_framework()` individually
- `lldb_setup_framework_rpaths_in_tool()` is obsolete; instead targets set them 
individually using `lldb_setup_rpaths` (old function will be removed in a 
separate commit to keep the diff readable)
- `lldb-framework` gets built by default
- the build-tree copy operation is a `POST_BUILD` command for the individual 
targets (instead of the `lldb-framework` target)
- while configuring the build, `lldb_framework_cleanup_list_raw.txt` collects 
`<command> <expr>` pairs to "strip" from the framework before install
- eventually, the final list of `<command> <path>` pairs is stored in 
`lldb_framework_cleanup_list_paths.txt`
- first action for `install` runs all the cleanup commands (see 
LLDBFrameworkInstall.cmake)

Test with monorepo:

  $ cmake -GNinja -DCMAKE_INSTALL_PREFIX=install/Developer/usr \
          -DLLDB_FRAMEWORK_INSTALL_DIR=install/SharedFrameworks \
          -DLLVM_TARGETS_TO_BUILD="X86;ARM;AArch64" 
-DLLVM_INSTALL_TOOLCHAIN_ONLY=ON \
          -DLLVM_ENABLE_PROJECTS="clang;libcxx;libcxxabi;lldb" \
          -DLLDB_BUILD_FRAMEWORK=ON -DLLDB_NO_INSTALL_DEFAULT_RPATH=ON 
../llvm-project/llvm
  $ ninja check-lldb
  $ ninja install

More background:

A framework is represented by the CMake target for the shared library it ships. 
Installing the target copies the entire bundle from the build-tree to the 
install-tree. This is mostly good, because we can add additional resource files 
to the build-tree framework bundle and `install` will include them 
automatically. We need the additional resources in the build-tree framework to 
facilitate testing.

Apart from simple files, however, LLDB.framework ships executable resources 
that define their own install targets (for extra processing at install-time 
like RPATH substitution or stripping). Apparently, CMake does not provide a way 
to run these install targets in the course of the framework install target. 
Instead they run either before or after framework install. The former is 
problematic, because the framework install will overwrite correctly processed 
resource executables with their build-tree pendants. While an install-order 
oriented fix may sound suitable at first appearance, we should keep in mind 
that, formally, CMake defines an order only locally 
<https://cmake.org/cmake/help/v3.4/command/install.html>:

  [install] This command generates installation rules for a project. Rules 
specified by calls to this command within a source directory are executed in 
order during installation. The order across directories is not defined.


Repository:
  rG LLVM Github Monorepo

https://reviews.llvm.org/D61952

Files:
  lldb/CMakeLists.txt
  lldb/cmake/modules/AddLLDB.cmake
  lldb/cmake/modules/LLDBConfig.cmake
  lldb/cmake/modules/LLDBFramework.cmake
  lldb/cmake/modules/LLDBFrameworkInstall.cmake
  lldb/tools/argdumper/CMakeLists.txt
  lldb/tools/darwin-debug/CMakeLists.txt
  lldb/tools/debugserver/source/CMakeLists.txt
  lldb/tools/driver/CMakeLists.txt
  lldb/tools/lldb-mi/CMakeLists.txt
  lldb/tools/lldb-server/CMakeLists.txt
  lldb/tools/lldb-vscode/CMakeLists.txt

Index: lldb/tools/lldb-vscode/CMakeLists.txt
===================================================================
--- lldb/tools/lldb-vscode/CMakeLists.txt
+++ lldb/tools/lldb-vscode/CMakeLists.txt
@@ -29,7 +29,3 @@
   LINK_COMPONENTS
     Support
   )
-
-if(LLDB_BUILD_FRAMEWORK)
-  lldb_setup_framework_rpaths_in_tool(lldb-vscode)
-endif()
Index: lldb/tools/lldb-server/CMakeLists.txt
===================================================================
--- lldb/tools/lldb-server/CMakeLists.txt
+++ lldb/tools/lldb-server/CMakeLists.txt
@@ -77,3 +77,7 @@
 )
 
 target_link_libraries(lldb-server PRIVATE ${LLDB_SYSTEM_LIBS})
+
+if(LLDB_BUILD_FRAMEWORK)
+  lldb_add_to_framework(lldb-server)
+endif()
Index: lldb/tools/lldb-mi/CMakeLists.txt
===================================================================
--- lldb/tools/lldb-mi/CMakeLists.txt
+++ lldb/tools/lldb-mi/CMakeLists.txt
@@ -93,7 +93,3 @@
   LINK_COMPONENTS
     Support
   )
-
-if(LLDB_BUILD_FRAMEWORK)
-  lldb_setup_framework_rpaths_in_tool(lldb-mi)
-endif()
Index: lldb/tools/driver/CMakeLists.txt
===================================================================
--- lldb/tools/driver/CMakeLists.txt
+++ lldb/tools/driver/CMakeLists.txt
@@ -24,5 +24,18 @@
 )
 
 if(LLDB_BUILD_FRAMEWORK)
-  lldb_setup_framework_rpaths_in_tool(lldb)
+  # Framework builds disable default install targets
+  install(TARGETS lldb COMPONENT lldb DESTINATION bin)
+
+  # In the build-tree, we know the exact path to the framework directory.
+  # The installed framework can be in different locations.
+  get_target_property(framework_build_dir liblldb LIBRARY_OUTPUT_DIRECTORY)
+  lldb_setup_rpaths(lldb
+    BUILD_RPATH
+      "${framework_build_dir}"
+    INSTALL_RPATH
+      "@loader_path/../../../SharedFrameworks"
+      "@loader_path/../../System/Library/PrivateFrameworks"
+      "@loader_path/../../Library/PrivateFrameworks"
+  )
 endif()
Index: lldb/tools/debugserver/source/CMakeLists.txt
===================================================================
--- lldb/tools/debugserver/source/CMakeLists.txt
+++ lldb/tools/debugserver/source/CMakeLists.txt
@@ -264,6 +264,10 @@
       ${entitlements}
     )
 
+  if(LLDB_BUILD_FRAMEWORK)
+    lldb_add_to_framework(debugserver)
+  endif()
+
   if(IOS)
     set_property(TARGET lldbDebugserverCommon APPEND PROPERTY COMPILE_DEFINITIONS
       WITH_LOCKDOWN
Index: lldb/tools/darwin-debug/CMakeLists.txt
===================================================================
--- lldb/tools/darwin-debug/CMakeLists.txt
+++ lldb/tools/darwin-debug/CMakeLists.txt
@@ -1,3 +1,7 @@
 add_lldb_tool(darwin-debug
   darwin-debug.cpp
   )
+
+if(LLDB_BUILD_FRAMEWORK)
+  lldb_add_to_framework(darwin-debug)
+endif()
Index: lldb/tools/argdumper/CMakeLists.txt
===================================================================
--- lldb/tools/argdumper/CMakeLists.txt
+++ lldb/tools/argdumper/CMakeLists.txt
@@ -4,3 +4,7 @@
   LINK_LIBS
     lldbUtility
   )
+
+if(LLDB_BUILD_FRAMEWORK)
+  lldb_add_to_framework(lldb-argdumper)
+endif()
Index: lldb/cmake/modules/LLDBFrameworkInstall.cmake
===================================================================
--- /dev/null
+++ lldb/cmake/modules/LLDBFrameworkInstall.cmake
@@ -0,0 +1,20 @@
+# If LLDB_BUILD_FRAMEWORK is enabled, this CMake script is included from cmake_install.cmake
+
+# Runs as first action of the install target, removing all files and directories
+# that were copied into the framework for testing.
+function(lldb_framework_prepare_install)
+  # Framework tools must install themselves, instead of being copied over inside
+  # the framework bundle. Affected files were added via lldb_add_to_framework()
+  # and logged in a text file in the build directory. Read all entries now.
+  file(STRINGS ${CMAKE_BINARY_DIR}/lldb_framework_cleanup_list_paths.txt cleanup_commands)
+
+  # Multiple cmake invocations may have added duplicates.
+  list(REMOVE_DUPLICATES cleanup_commands)
+
+  # Dump and delete each file/folder.
+  foreach(cmd ${cleanup_commands})
+    message(STATUS "Cleanup: ${cmd}")
+    separate_arguments(cmd)
+    execute_process(COMMAND ${CMAKE_COMMAND} -E ${cmd})
+  endforeach()
+endfunction()
Index: lldb/cmake/modules/LLDBFramework.cmake
===================================================================
--- lldb/cmake/modules/LLDBFramework.cmake
+++ lldb/cmake/modules/LLDBFramework.cmake
@@ -26,6 +26,11 @@
   MACOSX_FRAMEWORK_INFO_PLIST ${LLDB_SOURCE_DIR}/resources/LLDB-Info.plist.in
 )
 
+# Install in framework location
+install(TARGETS liblldb COMPONENT liblldb
+  FRAMEWORK DESTINATION ${LLDB_FRAMEWORK_INSTALL_DIR}
+)
+
 # Affects the layout of the framework bundle (default is macOS layout).
 if(IOS)
   set_target_properties(liblldb PROPERTIES
@@ -36,22 +41,9 @@
 endif()
 
 # Target to capture extra steps for a fully functional framework bundle.
-add_custom_target(lldb-framework)
+add_custom_target(lldb-framework ALL)
 add_dependencies(lldb-framework liblldb)
 
-# Dependencies are defined once tools are added (see AddLLDB.cmake)
-if(LLDB_FRAMEWORK_TOOLS)
-  message(STATUS "LLDB.framework: adding tools ${LLDB_FRAMEWORK_TOOLS}")
-  foreach(tool ${LLDB_FRAMEWORK_TOOLS})
-    add_custom_command(TARGET lldb-framework POST_BUILD
-      COMMAND ${CMAKE_COMMAND} -E copy $<TARGET_FILE:${tool}> $<TARGET_FILE_DIR:liblldb>/Resources
-      COMMENT "LLDB.framework: copy additional tool ${tool}"
-    )
-  endforeach()
-else()
-  message(WARNING "LLDB.framework: no additional tools configured (set via LLDB_FRAMEWORK_TOOLS)")
-endif()
-
 # Apart from this one, CMake creates all required symlinks in the framework bundle.
 add_custom_command(TARGET lldb-framework POST_BUILD
   COMMAND ${CMAKE_COMMAND} -E create_symlink
Index: lldb/cmake/modules/LLDBConfig.cmake
===================================================================
--- lldb/cmake/modules/LLDBConfig.cmake
+++ lldb/cmake/modules/LLDBConfig.cmake
@@ -64,8 +64,6 @@
   set(LLDB_FRAMEWORK_VERSION A CACHE STRING "LLDB.framework version (default is A)")
   set(LLDB_FRAMEWORK_BUILD_DIR bin CACHE STRING "Output directory for LLDB.framework")
   set(LLDB_FRAMEWORK_INSTALL_DIR Library/Frameworks CACHE STRING "Install directory for LLDB.framework")
-  set(LLDB_FRAMEWORK_TOOLS darwin-debug;debugserver;lldb-argdumper;lldb-server CACHE STRING
-      "List of tools to include in LLDB.framework/Resources")
 
   # Set designated directory for all dSYMs. Essentially, this emits the
   # framework's dSYM outside of the framework directory.
Index: lldb/cmake/modules/AddLLDB.cmake
===================================================================
--- lldb/cmake/modules/AddLLDB.cmake
+++ lldb/cmake/modules/AddLLDB.cmake
@@ -59,17 +59,9 @@
       ${pass_NO_INSTALL_RPATH}
     )
 
-    if (NOT LLVM_INSTALL_TOOLCHAIN_ONLY OR ${name} STREQUAL "liblldb")
+    if (NOT LLDB_BUILD_FRAMEWORK AND (NOT LLVM_INSTALL_TOOLCHAIN_ONLY OR ${name} STREQUAL "liblldb"))
       if (PARAM_SHARED)
-        if(${name} STREQUAL "liblldb" AND LLDB_BUILD_FRAMEWORK)
-          if(LLDB_FRAMEWORK_INSTALL_DIR)
-            set(install_dir ${LLDB_FRAMEWORK_INSTALL_DIR})
-          else()
-            set(install_dir ".")
-          endif()
-        else()
-          set(install_dir lib${LLVM_LIBDIR_SUFFIX})
-        endif()
+        set(install_dir lib${LLVM_LIBDIR_SUFFIX})
         install(TARGETS ${name}
           COMPONENT ${name}
           RUNTIME DESTINATION bin
@@ -129,7 +121,7 @@
   target_link_libraries(${name} PRIVATE ${ARG_LINK_LIBS})
   set_target_properties(${name} PROPERTIES FOLDER "lldb executables")
 
-  if(ARG_GENERATE_INSTALL)
+  if(ARG_GENERATE_INSTALL AND NOT LLDB_BUILD_FRAMEWORK)
     install(TARGETS ${name}
             COMPONENT ${name}
             RUNTIME DESTINATION bin)
@@ -204,3 +196,54 @@
 
   add_dependencies(${name} lldb-framework)
 endfunction()
+
+# Unified handling for executable LLDB.framework resources. Given the name of an
+# executable target, this function will: set the target's destination in the
+# install-tree, copy it to the framework in the build-tree and remove the copy
+# before installing the framework. The target's default build directory remains
+# unchanged.
+function(lldb_add_to_framework name)
+  # Same subdirectory in both cases, build-tree and install-tree.
+  set(subdir "LLDB.framework/Versions/${LLDB_FRAMEWORK_VERSION}/Resources")
+
+  # Destination for the copy in the build-tree. While the framework may not exist
+  # yet, it will exist when the generator expression gets expanded.
+  set(copy_dest "$<TARGET_FILE_DIR:liblldb>/../../../${subdir}")
+
+  # Copy into the framework's Resources directory for testing.
+  add_custom_command(TARGET ${name} POST_BUILD
+    COMMAND ${CMAKE_COMMAND} -E copy $<TARGET_FILE:${name}> ${copy_dest}
+    COMMENT "Copy ${name} to ${copy_dest}"
+  )
+
+  # Before installing, remove the copy from the build-tree framework, so it
+  # won't be installed together with the framework bundle and accidentally
+  # replace the target's already installed binary.
+  lldb_add_framework_cleanup_entry("remove ${copy_dest}/$<TARGET_FILE_NAME:${name}>")
+
+  # Install into the framework bundle in the install-tree.
+  install(TARGETS ${name} COMPONENT ${name}
+    DESTINATION ${LLDB_FRAMEWORK_INSTALL_DIR}/${subdir}
+  )
+endfunction()
+
+# Add an entry to the list of things we clean up before running install.
+# Generator expressions are expanded to actual paths at the end of the
+# configuration process. Entries consist of a command and a path,
+#  e.g.: remove /path/to/file
+#        remove_directory /path/to/directory
+function(lldb_add_framework_cleanup_entry entry)
+  file(APPEND ${CMAKE_BINARY_DIR}/lldb_framework_cleanup_list_raw.txt "${entry}\n")
+endfunction()
+
+# CMake's set_target_properties() doesn't allow to pass lists for RPATH
+# properties directly (error: "called with incorrect number of arguments").
+# Instead of defining two list variables each time, use this helper function.
+function(lldb_setup_rpaths name)
+  cmake_parse_arguments(LIST "" "" "BUILD_RPATH;INSTALL_RPATH" ${ARGN})
+  set_target_properties(${name} PROPERTIES
+    BUILD_WITH_INSTALL_RPATH OFF
+    BUILD_RPATH "${LIST_BUILD_RPATH}"
+    INSTALL_RPATH "${LIST_INSTALL_RPATH}"
+  )
+endfunction()
Index: lldb/CMakeLists.txt
===================================================================
--- lldb/CMakeLists.txt
+++ lldb/CMakeLists.txt
@@ -15,6 +15,13 @@
 include(LLDBConfig)
 include(AddLLDB)
 
+if(LLDB_BUILD_FRAMEWORK)
+  # Cleanup the build-tree before installing the framework. Install actions run
+  # in configure order. We add this early, so that cleanup runs first.
+  install(SCRIPT cmake/modules/LLDBFrameworkInstall.cmake
+          CODE "lldb_framework_prepare_install()")
+endif()
+
 # Define the LLDB_CONFIGURATION_xxx matching the build type
 if( uppercase_CMAKE_BUILD_TYPE STREQUAL "DEBUG" )
   add_definitions( -DLLDB_CONFIGURATION_DEBUG )
@@ -151,6 +158,10 @@
     list(APPEND LLDB_TEST_DEPS dsymutil)
   endif()
 
+  if(TARGET lldb-framework)
+    list(APPEND LLDB_TEST_DEPS lldb-framework)
+  endif()
+
   add_custom_target(lldb-test-deps)
   add_dependencies(lldb-test-deps ${LLDB_TEST_DEPS})
 
@@ -214,3 +225,13 @@
             COMMENT "Copying Python DLL to LLDB binaries directory.")
     endif ()
 endif ()
+
+if(LLDB_BUILD_FRAMEWORK)
+  # Expand generator expressions in the list of files to be removed from the
+  # framework bundle before installation. We run this late, because it requires
+  # the respective targets to exist.
+  file(GENERATE
+    OUTPUT ${CMAKE_BINARY_DIR}/lldb_framework_cleanup_list_paths.txt
+    INPUT ${CMAKE_BINARY_DIR}/lldb_framework_cleanup_list_raw.txt
+  )
+endif()
_______________________________________________
lldb-commits mailing list
lldb-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits

Reply via email to