blob: 8b9627911a818357159fcd7f2a7c2ad18fdb684e [file] [log] [blame]
# **********************************************************
# Copyright (c) 2010-2024 Google, Inc. All rights reserved.
# Copyright (c) 2010 VMware, Inc. All rights reserved.
# **********************************************************
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
#
# * Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# * Neither the name of VMware, Inc. nor the names of its contributors may be
# used to endorse or promote products derived from this software without
# specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL VMWARE, INC. OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
# DAMAGE.
# symbol access library
cmake_minimum_required(VERSION 3.7)
include(../../make/policies.cmake NO_POLICY_SCOPE)
if (X64)
set(BITS "64")
else ()
set(BITS "32")
endif ()
if (DR_HOST_ARM)
if (ANDROID)
set(ARCH "-android")
else ()
set(ARCH "-arm")
# XXX i#1501: to support both arm-linux-gnueabi and arm-linux-gnueabi,
# we rely on CMAKE_C_LIBRARY_ARCHITECTURE for libelftc libraries selection.
# If it is not set by some system, users need manually set it to gnueabi
# for using gnueabi build of libelftc libraries.
if (CMAKE_C_LIBRARY_ARCHITECTURE MATCHES "gnueabi$")
set(BITS "${BITS}-eabi")
else ()
set(BITS "${BITS}-eabihf")
endif ()
endif ()
endif()
if (DR_HOST_AARCH64)
set(ARCH "-aarch64")
endif ()
if (DR_HOST_RISCV64)
set(ARCH "-riscv64")
endif ()
# we need libc b/c our elftoolchain libraries use it
set(DynamoRIO_USE_LIBC ON)
set(USE_ELFUTILS OFF)
# We use our own .lib file to support VS2005 whose dbghelp.lib doesn't have some
# routines we want to use.
if (WIN32)
# XXX: if we add any more of these .lib files we should share this code
# (currently we have make/ntdll_imports.cmake and here).
find_program(LIB_EXECUTABLE lib.exe HINTS "${cl_path}" DOC "path to lib.exe")
if (NOT LIB_EXECUTABLE)
message(FATAL_ERROR "Cannot find lib.exe")
endif (NOT LIB_EXECUTABLE)
set(dbghelp_src "${CMAKE_CURRENT_SOURCE_DIR}/dbghelp_imports.c")
set(dbghelp_def "${CMAKE_CURRENT_SOURCE_DIR}/dbghelp_imports.def")
# We need a different name so we can also use the VS dbghelp.lib
set(dbghelp_lib "${CMAKE_CURRENT_BINARY_DIR}/dbghelp_imports.lib")
set_property(SOURCE "${dbghelp_lib}" PROPERTY GENERATED true)
# Because the exports are stdcall we can't just use a .def file: we need
# an .obj file built from stubs w/ the same signatures.
# We don't need a stamp file b/c the .lib is not a source; plus a stamp
# file causes Ninja to fail to build.
add_custom_command(OUTPUT "${dbghelp_lib}"
DEPENDS "${dbghelp_src}" "${dbghelp_def}"
COMMAND "${CMAKE_C_COMPILER}" ARGS /nologo /c /Ob0 ${dbghelp_src}
COMMAND "${LIB_EXECUTABLE}" ARGS
/nologo /name:dbghelp.dll /def:${dbghelp_def}
${CMAKE_CURRENT_BINARY_DIR}/dbghelp_imports.obj
WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
VERBATIM # recommended: p260 of cmake book
)
if ("${CMAKE_GENERATOR}" MATCHES "Visual Studio")
set(dbghelp_dep "")
# for correct parallel builds we need a target to avoid
# duplicate and racy VS per-project rules
add_custom_target(dbghelp_tgt DEPENDS "${dbghelp_lib}")
else ()
set(dbghelp_dep "${dbghelp_lib}")
endif ()
set(dbghelp_flags "${dbghelp_lib}")
set(srcs
drsyms_windows.c drsyms_unix_common.c drsyms_pecoff.c
drsyms_dwarf.c demangle.cc drsyms_common.c
${dbghelp_dep})
# i#1491#2: VS generators fail if static lib has resources
set(srcs_static ${srcs})
set(srcs ${srcs} ${PROJECT_SOURCE_DIR}/core/win32/resources.rc)
set(dwarf_dir "${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc-pecoff/lib${BITS}")
set(dwarf_libpath "${dwarf_dir}/dwarf.lib")
set(elftc_libpath "${dwarf_dir}/elftc.lib")
configure_file("${dwarf_dir}/dwarf.pdb"
"${CMAKE_LIBRARY_OUTPUT_DIRECTORY}/dwarf.pdb" COPYONLY)
configure_file("${dwarf_dir}/elftc.pdb"
"${CMAKE_LIBRARY_OUTPUT_DIRECTORY}/elftc.pdb" COPYONLY)
elseif (UNIX)
set(srcs
drsyms_unix_frontend.c drsyms_unix_common.c
demangle.cc drsyms_common.c)
if (APPLE)
set(srcs ${srcs} drsyms_dwarf.c drsyms_macho.c)
set(dwarf_libpath
"${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc-macho${ARCH}/lib${BITS}/libdwarf.a")
set(elftc_libpath
"${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc-macho${ARCH}/lib${BITS}/libelftc.a")
elseif (ANDROID)
# TODO i#5926: Use elfutils for Android. First we need to get zlib installed
# in our test environment.
set(srcs ${srcs} drsyms_dwarf.c drsyms_elf.c)
set(dwarf_libpath
"${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc${ARCH}/lib${BITS}/libdwarf.a")
set(elftc_libpath
"${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc${ARCH}/lib${BITS}/libelftc.a")
set(elf_libpath
"${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc${ARCH}/lib${BITS}/libelf.a")
else ()
set(elftc_libpath
"${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc${ARCH}/lib${BITS}/libelftc.a")
message(STATUS "Using elfutils")
# TODO i#5926: Use elfutils everywhere. We start out with just Linux.
set(USE_ELFUTILS ON)
set(elfutils_dir "${PROJECT_SOURCE_DIR}/third_party/elfutils")
if (NOT EXISTS "${elfutils_dir}")
message(FATAL_ERROR "Missing required submodule ${elfutils_dir}")
endif ()
# Apply patches.
# XXX i#5926: Better to fork the elfutils repo and apply these in the fork
# and have the fork be the source of our submodule? We could store config.h
# there too.
find_program(PATCH patch DOC "patch")
if (NOT PATCH)
message(FATAL_ERROR "Unable to find patch")
endif ()
file(GLOB patches "${CMAKE_CURRENT_SOURCE_DIR}/elfutils/*.patch")
foreach (patch ${patches})
get_filename_component(patch_base ${patch} NAME)
string(REGEX REPLACE ".patch$" "" patch_base ${patch_base})
file(GLOB orig_path "${elfutils_dir}/*/${patch_base}")
list(LENGTH orig_path glob_count)
if (NOT glob_count EQUAL 1)
message(FATAL_ERROR "Failed to find single source for ${patch}")
endif ()
list(APPEND patch_srcs ${orig_path})
set(patch_path "${CMAKE_CURRENT_BINARY_DIR}/${patch_base}")
execute_process(COMMAND ${PATCH} -p1 -d "${elfutils_dir}" -o "${patch_path}"
INPUT_FILE "${patch}"
RESULT_VARIABLE patch_result ERROR_VARIABLE patch_err)
if (patch_result)
message(FATAL_ERROR "Failed to apply ${patch}: ${patch_err}")
endif ()
endforeach ()
# Add the elfutils library build rules we need. We want PIC static libs.
foreach (lib elf;dw;dwelf;ebl)
file(GLOB ${lib}_files "${elfutils_dir}/lib${lib}/*.c")
foreach (orig_path ${patch_srcs})
if ("${orig_path}" IN_LIST ${lib}_files)
# Swap in our patched file.
list(REMOVE_ITEM ${lib}_files "${orig_path}")
get_filename_component(base ${orig_path} NAME)
set(patch_path "${CMAKE_CURRENT_BINARY_DIR}/${base}")
list(APPEND ${lib}_files "${patch_path}")
message(STATUS "Swapped in patched ${patch_path}")
endif ()
endforeach ()
add_library(${lib}_pic STATIC ${${lib}_files})
# We want to directly use DR's allocator instead of relying on its private loader
# redirecting in order to support static usage with no loader.
# ld is not actually used, so we can't use its -wrap=malloc feature.
# Instead we rely on the preprocessor.
set_target_properties(${lib}_pic PROPERTIES
# We have a presumably-widely-applicable config.h in drsyms/elfutils.
INCLUDE_DIRECTORIES
"${CMAKE_CURRENT_SOURCE_DIR}/elfutils;${elfutils_dir}/lib;${elfutils_dir}/libasm;${elfutils_dir}/libebl;${elfutils_dir}/libdwelf;${elfutils_dir}/libdwfl"
COMPILE_DEFINITIONS
"_GNU_SOURCE;HAVE_CONFIG_H;_FORTIFY_SOURCE=3;PIC;SHARED;SYMBOL_VERSIONING;malloc=__wrap_malloc;calloc=__wrap_calloc;realloc=__wrap_realloc;free=__wrap_free;strdup=__wrap_strdup"
COMPILE_FLAGS "-std=gnu99 -Wall -g -O2 -fPIC")
DR_export_target(${lib}_pic)
install_exported_target(${lib}_pic ${INSTALL_EXT_LIB})
copy_target_to_device(${lib}_pic "${location_suffix}")
endforeach ()
# libdw uses pthread_rwlock_* routines.
link_with_pthread(dw_pic)
include_directories("${elfutils_dir}/libelf")
include_directories("${elfutils_dir}/libdw")
set(srcs ${srcs} drsyms_dw.c drsyms_elf.c)
add_definitions(-DUSE_ELFUTILS)
if (ZLIB_FOUND)
add_definitions(-DHAS_ZLIB)
include_directories(${ZLIB_INCLUDE_DIRS})
else ()
message(FATAL_ERROR "zlib not found but required to build drsyms_static on Linux")
endif ()
# Avoid stdbool.h from libdw.h defining _Bool after dr_defines.h uses char.
add_definitions(-DDR__Bool_EXISTS)
endif ()
set(srcs_static ${srcs})
endif (WIN32)
# while private loader means preferred base is not required, more efficient
# to avoid rebase so we avoid conflict w/ client and other exts
set(PREFERRED_BASE 0x76000000)
add_library(drsyms SHARED ${srcs})
configure_extension(drsyms OFF)
macro(configure_drsyms_target target)
if (WIN32)
target_link_libraries(${target} dbghelp ${dbghelp_flags})
if ("${CMAKE_GENERATOR}" MATCHES "Visual Studio")
# for parallel build correctness we need a target dependence
add_dependencies(${target} dbghelp_tgt)
endif ()
endif ()
# we always use the elftoolchain library when building with cmake
append_property_list(TARGET ${target} COMPILE_DEFINITIONS "DRSYM_HAVE_LIBELFTC")
# For elfutils the linking is the same for shared and static so we place here.
if (USE_ELFUTILS)
target_link_libraries(${target} dw_pic)
if (LINUX)
target_link_libraries(${target} dwelf_pic elf_pic)
endif ()
target_link_libraries(${target} ebl_pic ${ZLIB_LIBRARIES})
endif ()
endmacro(configure_drsyms_target)
configure_drsyms_target(drsyms)
use_DynamoRIO_extension(drsyms drcontainers)
include_directories("${PROJECT_SOURCE_DIR}/ext/drsyms/libelftc/include")
# We require DynamoRIO_USE_LIBC, as there's no simple automated
# solution to get clients using a static drsyms to omit /noentry and
# link with libcmt.lib prior to linking w/ DR, unless we change
# configure_DynamoRIO_client() to take in a list of all extensions
# that will be used -- but we now have DynamoRIO_USE_LIBC ON by
# default.
# Listing the dependent libs as sources does get them combined into
# drsyms static lib for pre-VS2010, but VS2010 won't do that, so we go
# with exported separate libs to match Linux (alternative would be a custom
# command to combine via lib.exe).
add_library(drsyms_static STATIC ${srcs_static})
configure_extension(drsyms_static ON)
configure_drsyms_target(drsyms_static)
use_DynamoRIO_extension(drsyms_static drcontainers)
if (NOT USE_ELFUTILS)
target_link_libraries(drsyms dwarf)
if (LINUX)
target_link_libraries(drsyms elf)
endif ()
endif ()
target_link_libraries(drsyms elftc)
# i#693: CMake will try to export the path to the static libs we use via
# IMPORTED_LINK_INTERFACE_LIBRARIES_NOCONFIG, but they won't exist on the
# user's machine. Clearing this property prevents that.
# i#3474: LINK_INTERFACE_LIBRARIES is deprecated, which is overridden by
# INTERFACE_LINK_LIBRARIES property if policy CMP0022 is NEW.
set_target_properties(drsyms PROPERTIES INTERFACE_LINK_LIBRARIES "")
# If drsyms is built static we need to include libelftc libs with an exports path
# in DynamoRIOTarget*.cmake and not with the source path here:
if (NOT USE_ELFUTILS)
add_library(dwarf STATIC IMPORTED)
set_property(TARGET dwarf PROPERTY IMPORTED_LOCATION "${dwarf_libpath}")
target_link_libraries(drsyms_static dwarf)
if (LINUX)
add_library(elf STATIC IMPORTED)
set_property(TARGET elf PROPERTY IMPORTED_LOCATION "${elf_libpath}")
target_link_libraries(drsyms_static elf)
endif (LINUX)
endif ()
add_library(elftc STATIC IMPORTED)
set_property(TARGET elftc PROPERTY IMPORTED_LOCATION "${elftc_libpath}")
target_link_libraries(drsyms_static elftc)
if (UNIX)
# Avoid missing symbols in static library build from g++ libs when
# drsyms_bench is linked with gcc instead of g++ (i#715, happens w/ cmake
# < 2.8.0 where demangle.cc is not propagated through libdrsyms.a)
append_property_string(TARGET drsyms_static COMPILE_FLAGS "-fno-exceptions")
# On ARM cross-compilation I'm seeing the same __gxx_personality_v0 and
# __cxa_end_cleanup errors, so we pass this for the .so as well:
append_property_string(TARGET drsyms COMPILE_FLAGS "-fno-exceptions")
endif (UNIX)
if (WIN32)
append_property_string(TARGET drsyms_static COMPILE_FLAGS "/DSTATIC_LIB")
endif (WIN32)
DR_install(FILES "${dwarf_libpath}" "${elf_libpath}" "${elftc_libpath}"
DESTINATION ${INSTALL_EXT_LIB})
if (WIN32)
DR_install(FILES "${dwarf_dir}/dwarf.pdb" "${dwarf_dir}/elftc.pdb"
DESTINATION ${INSTALL_EXT_LIB}
PERMISSIONS ${owner_access} OWNER_EXECUTE GROUP_READ GROUP_EXECUTE
WORLD_READ WORLD_EXECUTE)
endif (WIN32)
add_executable(drsyms_bench drsyms_bench.c)
configure_DynamoRIO_standalone(drsyms_bench)
use_DynamoRIO_extension(drsyms_bench drsyms)
# we don't want drsyms_bench installed so we avoid the standard location
set_target_properties(drsyms_bench PROPERTIES
RUNTIME_OUTPUT_DIRECTORY${location_suffix} "${PROJECT_BINARY_DIR}/ext")
# documentation is put into main DR docs/ dir
install_ext_header(drsyms.h)
if (WIN32)
# We need separate 32-bit and 64-bit versions so we put into lib dir.
# We put into base dir and not ext for easy sharing w/ including clients.
DR_install(FILES ${dbghelp_lib} DESTINATION ${INSTALL_LIB_BASE})
# i#1344: Include a copy of a recent (> 6.0) redistributable version of dbghelp,
# if we found one, so our drsyms works pre-Vista.
if (dbghelp_path)
configure_file(${dbghelp_path} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY}/dbghelp.dll
COPYONLY)
DR_install(FILES "${dbghelp_path}" DESTINATION ${INSTALL_EXT_LIB})
endif ()
endif (WIN32)