build: Fix linking for fuzz target when building with MSan #30778

pull hebasto wants to merge 2 commits into bitcoin:master from hebasto:240831-ci-msan changing 3 files +13 −1
  1. hebasto commented at 6:12 am on August 31, 2024: member

    The first commit fixes #30760.

    The second commit:

    1. Preserves -g -O1 set in MSAN_FLAGS. Since configuration-specific flags override general flags, these are set to empty strings. A similar approach is used in the OSS-Fuzz repository.
    2. Sets the “Debug” build configuration when depends are built with DEBUG=1, ensuring that linux_debug_CPPFLAGS from depends are passed to the main build system.
  2. DrahtBot commented at 6:12 am on August 31, 2024: contributor

    The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.

    Code Coverage

    For detailed information about the code coverage, see the test coverage report.

    Reviews

    See the guideline for information on the review process.

    Type Reviewers
    ACK maflcko, fanquake

    If your review is incorrectly listed, please react with 👎 to this comment and the bot will ignore it on the next update.

    Conflicts

    Reviewers, this pull request conflicts with the following ones:

    • #28690 (build: Introduce internal kernel library by TheCharlatan)

    If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.

  3. DrahtBot added the label Tests on Aug 31, 2024
  4. hebasto force-pushed on Aug 31, 2024
  5. DrahtBot added the label CI failed on Aug 31, 2024
  6. DrahtBot commented at 6:47 am on August 31, 2024: contributor

    🚧 At least one of the CI tasks failed. Debug: https://github.com/bitcoin/bitcoin/runs/29506147599

    Make sure to run all tests locally, according to the documentation.

    The failure may happen due to a number of reasons, for example:

    • Possibly due to a silent merge conflict (the changes in this pull request being incompatible with the current code in the target branch). If so, make sure to rebase on the latest commit of the target branch.

    • A sanitizer issue, which can only be found by compiling with the sanitizer and running the affected test.

    • An intermittent issue.

    Leave a comment here, if you need help tracking down a confusing failure.

  7. hebasto closed this on Aug 31, 2024

  8. hebasto reopened this on Aug 31, 2024

  9. hebasto force-pushed on Aug 31, 2024
  10. in ci/test/00_setup_env_native_fuzz_with_msan.sh:23 in 923f96e9a6 outdated
    16@@ -17,8 +17,12 @@ export PACKAGES="ninja-build"
    17 # BDB generates false-positives and will be removed in future
    18 export DEP_OPTS="DEBUG=1 NO_BDB=1 NO_QT=1 CC=clang CXX=clang++ CFLAGS='${MSAN_FLAGS}' CXXFLAGS='${MSAN_AND_LIBCXX_FLAGS}'"
    19 export GOAL="install"
    20+# Setting CMAKE_{C,CXX}_FLAGS_DEBUG flags to an empty string ensures that the flags set in MSAN_FLAGS remain unaltered.
    21 # _FORTIFY_SOURCE is not compatible with MSAN.
    22 export BITCOIN_CONFIG="\
    23+ -DCMAKE_BUILD_TYPE="Debug" \
    


    fanquake commented at 8:37 am on August 31, 2024:
    If setting the build type specific flags to empty is all that’s needed, why are you also changing the build type to Debug, rather than leaving it as RelwithDebInfo (same as oss-fuzz).

    hebasto commented at 8:39 am on August 31, 2024:

    To pass linux_debug_CPPFLAGS from depends.

    (I was going to add this explanation to the PR description after more tests completed).

  11. hebasto force-pushed on Aug 31, 2024
  12. hebasto force-pushed on Aug 31, 2024
  13. DrahtBot removed the label CI failed on Aug 31, 2024
  14. maflcko commented at 6:48 am on September 2, 2024: member
    review-only ACK 331e9761b639f380c457582ae3c4b05e56ff02b0
  15. hebasto commented at 7:03 am on September 2, 2024: member

    review-only ACK 331e9761b639f380c457582ae3c4b05e56ff02b0

    I’m still working on it. MSan fuzzing CI job still fails when running locally.

  16. build: Fix linking for `fuzz` target when building with MSan 26c460aa8b
  17. ci: Do not override `-g -O1` set in `MSAN_FLAGS`
    Additionally, setting the "Debug" build configuration ensures that
    `linux_debug_CPPFLAGS` from depends are passed to the main build system.
    787dfaf084
  18. hebasto force-pushed on Sep 2, 2024
  19. hebasto renamed this:
    ci: Do not override `-g -O1` set in `MSAN_FLAGS`
    build: Fix linking for `fuzz` target when building with MSan
    on Sep 2, 2024
  20. hebasto marked this as ready for review on Sep 2, 2024
  21. hebasto commented at 10:59 pm on September 2, 2024: member

    review-only ACK 331e976

    I’m still working on it. MSan fuzzing CI job still fails when running locally.

    The issue has been resolved. The PR description has been updated.

    Ready for review and testing.

  22. DrahtBot added the label CI failed on Sep 3, 2024
  23. DrahtBot removed the label CI failed on Sep 3, 2024
  24. in src/test/fuzz/CMakeLists.txt:135 in 26c460aa8b outdated
    131@@ -132,6 +132,7 @@ target_link_libraries(fuzz
    132   test_fuzz
    133   bitcoin_cli
    134   bitcoin_common
    135+  bitcoin_util
    


    maflcko commented at 6:26 am on September 3, 2024:

    26c460aa8b5decfd08d931b9b3f80be5c13c7528:

    Can you explain this a bit better? Why did it only fail on one CI config, and none of the others? Why was the error an msan runtime-error, as opposed to a link-time error?


    hebasto commented at 7:52 am on September 3, 2024:

    26c460a:

    Can you explain this a bit better?

    I’m afraid I can’t (

    Why did it only fail on one CI config, and none of the others? Why was the error an msan runtime-error, as opposed to a link-time error?

    Those are the questions I asked myself :)

    I’m not able to answer them at this moment. There are two reasons for that:

    1. The current set of internal libraries, including test_fuzz and test_util, has circular symbol dependencies, which is inherited from Autotools.
    2. I’m not fully aware of how the linker works with -fsanitize=fuzzer,memory.

    maflcko commented at 8:35 am on September 3, 2024:
    Ok, so I guess the reason is that this is required to mirror the autotools approach from https://github.com/bitcoin/bitcoin/blob/5abb9b1af49be9024f95fa2f777285c531785d85/src/Makefile.test.include#L44 ?

    hebasto commented at 8:45 am on September 3, 2024:

    Autotools tends to overlink, while in CMake, I aimed to link only the necessary components.

    Yes, this PR commit mirrors the line from bitcoin/src/Makefile.test.include that you mentioned. However, simply removing that line from 80f00cafdeef0600fa1a5e906686720786c2336c results in a link-time error.

  25. fanquake commented at 8:45 am on September 3, 2024: member

    Sets the “Debug” build configuration when depends are built with DEBUG=1, ensuring that linux_debug_CPPFLAGS from depends are passed to the main build system.

    Is this not a bug? If depends is built (in any configuration/with any options/flags) then the expectation would be that everything is passed to CMake (regardless of the CMake build type). However it seems like your saying CMake will actually just ignore some flags depending on which build type has been set? If this was the intended behaviour, has this been documented somewhere? Otherwise how are builders meant to know that certains flags from depends might just be ignored, depending on which build type is selected in CMake?

  26. maflcko commented at 9:24 am on September 3, 2024: member

    review-only ACK 787dfaf084a3952319778da9cbcda9d7d619e4ee

    Only change since last review is linking with libbitcoin_util.

    It would be nice to better understand this change, but this can be done later and is not a blocker to for a test-only change.

    Also, the depends behavior can be documented better, or be changed, but this is probably best done in a separate issue or pull request, so that this test-only change stays focussed.

  27. hebasto commented at 10:16 am on September 3, 2024: member

    Sets the “Debug” build configuration when depends are built with DEBUG=1, ensuring that linux_debug_CPPFLAGS from depends are passed to the main build system.

    Is this not a bug?

    No, it is not. Rather it is a design feature.

    If depends is built (in any configuration/with any options/flags) then the expectation would be that everything is passed to CMake (regardless of the CMake build type).

    While both depends and CMake have configuration-specific flags, it seems unreasonable to expect from CMake configured for the “Release” build type pull in *_debug_* flags from depends.

    If this was the intended behaviour, has this been documented somewhere?

    Yes, it was intended. No, it has not been documented. FWIW, the fact that flags from depends override flags set in the main build system has also never been documented.


    Here are more details regarding that design decision (C++ flags are used as an example, but the same applies for other tool flags):

    1. CMake has (a) flags applied for all build configurations (CMAKE_CXX_FLAGS) and (b) configuration-specific flags (CMAKE_CXX_FLAGS_<CONFIG>). The latter follow the former during the build tool invocation.

    2. The depends build subsystem has a similar design, and configuration-specific variables have _release_ and _debug_ infixes. The depends build configuration is determined by whether DEBUG variable is defined as 1 or not.

    3. IMPORTANT note about behaviour from Autotools: Flags set in depends override ones set in the main build system.

    4. The CMake’s way of passing the toolchain details set in depends to the main build system is by using a toolchain file. There are dedicated CMake variables to be defined in the toolchain file, namely CMAKE_CXX_FLAGS_INIT and configuration-specific CMAKE_CXX_FLAGS_<CONFIG>_INIT. The former is assigned the host_CXXFLAGS value, and the latter use host_release_CXXFLAGS and host_debug_CXXFLAGS values. This helps to ensure that the behaviour described in item 3 still holds when using CMake.

  28. fanquake commented at 10:28 am on September 3, 2024: member

    it seems unreasonable to expect from CMake configured for the “Release” build type pull in debug flags from depends.

    CMake shouldn’t have a concept of “depends debug” flags (it shouldn’t really even know about depends). It should just get flags/options passed to it in a toolchain file (which is the interface between the two), and it should use them. Having any special coupling/behaviour based on if things are a “depends” build or not, removes the ability for anyone else to bring a CMake toolchain file, and expect it to work properly.

    We should open an issue to follow up on this, as at a minimum, it needs better documentation around expections / how things are meant to work, or even reworking, as at the moment, the behaviour seems inconsistent/unexpected.

  29. fanquake approved
  30. fanquake commented at 10:58 am on September 3, 2024: member

    ACK 787dfaf084a3952319778da9cbcda9d7d619e4ee - as a follow up it would be good to:

    • Actually figure out why this fix works, and why the other msan build wasn’t broken: #30778 (review).
    • Update oss-fuzz to match what is being done here? Now that they’ve diverged.
    • Open an issue to figure out expectations around depends-with-cmake behaviour.
  31. maflcko commented at 11:00 am on September 3, 2024: member

    For reference, before (https://cirrus-ci.com/task/6668539271053312?logs=ci#L1064):

    0Cross compiling ....................... FALSE
    1C++ compiler .......................... Clang 18.1.3, /usr/bin/clang++
    2CMAKE_BUILD_TYPE ...................... RelWithDebInfo
    3Preprocessor defined macros ........... 
    4C++ compiler flags .................... -fsanitize=memory -fsanitize-memory-track-origins=2 -fno-omit-frame-pointer -g -O1 -fno-optimize-sibling-calls -nostdinc++ -nostdlib++ -isystem /msan/cxx_build/include/c++/v1 -L/msan/cxx_build/lib -Wl,-rpath,/msan/cxx_build/lib -lc++ -lc++abi -lpthread -Wno-unused-command-line-argument -O2 -g -std=c++20 -fPIC -fdebug-prefix-map=/ci_container_base=. -fmacro-prefix-map=/ci_container_base=. -Werror -fsanitize=memory -Wall -Wextra -Wgnu -Wformat -Wformat-security -Wvla -Wshadow-field -Wthread-safety -Wloop-analysis -Wredundant-decls -Wunused-member-function -Wdate-time -Wconditional-uninitialized -Woverloaded-virtual -Wsuggest-override -Wimplicit-fallthrough -Wunreachable-code -Wdocumentation -Wself-assign -Wundef -Wno-unused-parameter -U_FORTIFY_SOURCE -D_FORTIFY_SOURCE=3 -Wstack-protector -fstack-protector-all -fcf-protection=full -fstack-clash-protection -U_FORTIFY_SOURCE
    5Linker flags .......................... -fsanitize=memory -fsanitize-memory-track-origins=2 -fno-omit-frame-pointer -g -O1 -fno-optimize-sibling-calls -nostdinc++ -nostdlib++ -isystem /msan/cxx_build/include/c++/v1 -L/msan/cxx_build/lib -Wl,-rpath,/msan/cxx_build/lib -lc++ -lc++abi -lpthread -Wno-unused-command-line-argument -O2 -g -fsanitize=memory -fstack-protector-all -fcf-protection=full -fstack-clash-protection -Wl,-z,relro -Wl,-z,now -Wl,-z,separate-code -fPIE -pie
    

    After (https://api.cirrus-ci.com/v1/task/6661080322146304/logs/ci.log):

    0Cross compiling ....................... FALSE
    1C++ compiler .......................... Clang 18.1.3, /usr/bin/clang++
    2CMAKE_BUILD_TYPE ...................... Debug
    3Preprocessor defined macros ........... DEBUG DEBUG_LOCKORDER DEBUG_LOCKCONTENTION RPC_DOC_CHECK ABORT_ON_FAILED_ASSUME _GLIBCXX_DEBUG _GLIBCXX_DEBUG_PEDANTIC _LIBCPP_HARDENING_MODE=_LIBCPP_HARDENING_MODE_DEBUG
    4C++ compiler flags .................... -fsanitize=memory -fsanitize-memory-track-origins=2 -fno-omit-frame-pointer -g -O1 -fno-optimize-sibling-calls -nostdinc++ -nostdlib++ -isystem /msan/cxx_build/include/c++/v1 -L/msan/cxx_build/lib -Wl,-rpath,/msan/cxx_build/lib -lc++ -lc++abi -lpthread -Wno-unused-command-line-argument -std=c++20 -fPIC -fdebug-prefix-map=/ci_container_base=. -fmacro-prefix-map=/ci_container_base=. -Werror -fsanitize=memory -Wall -Wextra -Wgnu -Wformat -Wformat-security -Wvla -Wshadow-field -Wthread-safety -Wloop-analysis -Wredundant-decls -Wunused-member-function -Wdate-time -Wconditional-uninitialized -Woverloaded-virtual -Wsuggest-override -Wimplicit-fallthrough -Wunreachable-code -Wdocumentation -Wself-assign -Wundef -Wno-unused-parameter -Wstack-protector -fstack-protector-all -fcf-protection=full -fstack-clash-protection -U_FORTIFY_SOURCE
    5Linker flags .......................... -fsanitize=memory -fsanitize-memory-track-origins=2 -fno-omit-frame-pointer -g -O1 -fno-optimize-sibling-calls -nostdinc++ -nostdlib++ -isystem /msan/cxx_build/include/c++/v1 -L/msan/cxx_build/lib -Wl,-rpath,/msan/cxx_build/lib -lc++ -lc++abi -lpthread -Wno-unused-command-line-argument -fsanitize=memory -fstack-protector-all -fcf-protection=full -fstack-clash-protection -Wl,-z,relro -Wl,-z,now -Wl,-z,separate-code -fPIE -pie
    
  32. fanquake merged this on Sep 3, 2024
  33. fanquake closed this on Sep 3, 2024

  34. hebasto deleted the branch on Sep 3, 2024
  35. hebasto commented at 12:52 pm on September 3, 2024: member

    … as a follow up it would be good to:

    • Update oss-fuzz to match what is being done here? Now that they’ve diverged.

    The OSS-Fuzz script controls all flags explicitly, including https://github.com/google/oss-fuzz/blob/5403c34ff006baf410b94cad1112a6c385d1bade/projects/bitcoin-core/build.sh#L50:

    0export CPPFLAGS="-D_LIBCPP_HARDENING_MODE=_LIBCPP_HARDENING_MODE_DEBUG -DBOOST_MULTI_INDEX_ENABLE_SAFE_MODE"
    
  36. fanquake commented at 12:57 pm on September 3, 2024: member

    The OSS-Fuzz script controls all flags explicitly

    Sure, but you changed the CMake build type here, so why not change it in oss-fuzz? Otherwise, it’d be good to explain somewhere why they (should) differ.

  37. hebasto commented at 1:11 pm on September 3, 2024: member

    The OSS-Fuzz script controls all flags explicitly

    Sure, but you changed the CMake build type here, so why not change it in oss-fuzz? Otherwise, it’d be good to explain somewhere why they (should) differ.

    Sure. Please consider https://github.com/google/oss-fuzz/pull/12443.

  38. hebasto commented at 2:07 pm on September 3, 2024: member
  39. in ci/test/00_setup_env_native_fuzz_with_msan.sh:20 in 787dfaf084
    16@@ -17,8 +17,12 @@ export PACKAGES="ninja-build"
    17 # BDB generates false-positives and will be removed in future
    18 export DEP_OPTS="DEBUG=1 NO_BDB=1 NO_QT=1 CC=clang CXX=clang++ CFLAGS='${MSAN_FLAGS}' CXXFLAGS='${MSAN_AND_LIBCXX_FLAGS}'"
    19 export GOAL="install"
    20+# Setting CMAKE_{C,CXX}_FLAGS_DEBUG flags to an empty string ensures that the flags set in MSAN_FLAGS remain unaltered.
    


    ryanofsky commented at 3:19 pm on September 3, 2024:
    I think I understand why this needs to pass BUILD_TYPE=Debug so the debug flags from the depends system will be applied. But I don’t understand why FLAGS_DEBUG variables need to be set to empty stings. How does setting those to empty strings cause MSAN_FLAGS to remain unaltered? What code is altering MSAN_FLAGS if FLAGS_DEBUG variables are unset?

    hebasto commented at 3:26 pm on September 3, 2024:
    Perhaps I used the word ‘unaltered’ incorrectly. It would be more accurate to say that “flags set in MSAN_FLAGS won’t be overridden”. The CMAKE_CXX_FLAGS_DEBUG follow CMAKE_CXX_FLAGS in the compiler invocation string and competing flags may be overridden.

    ryanofsky commented at 3:42 pm on September 3, 2024:

    Good point that “overridden” is probably more appropriate word, but I think I already understood that this meant overridden.

    The thing I don’t understand is what causes MSAN_FLAGS to be overridden when FLAGS_DEBUG variables are unset, but not overridden when they are empty? Appreciate the clarification. I’m not asking for any particular reason, just curious to understand this better.


    hebasto commented at 4:29 pm on September 3, 2024:

    Sure. The logic is follows:

    1. The MSAN_FLAGS variable’s value is ... -g -O1 ....
    2. This value is assigned to CXXFLAGS for building depends and become the value of CMAKE_CXX_FLAGS in the toolchain file.
    3. Being configured with -DCMAKE_BUILD_TYPE=Debug, the build system may assign -O0 -ftrapv -g3 to CMAKE_CXX_FLAGS_DEBUG.
    4. When invoking a compier, CMAKE_CXX_FLAGS and CMAKE_CXX_FLAGS_DEBUG are concatenated, resulting in .. -g -O1 ... -O0 ... -g3 .... This means that flags set in MSAN_FLAGS will not have any effect.

    ryanofsky commented at 5:07 pm on September 3, 2024:
    Very helpful, thanks. I think I would write comment as “Set CMAKE_{C,CXX}_FLAGS_DEBUG options to empty so CMake’s default options for debug builds do not interfere with MSAN flags”

    ryanofsky commented at 4:37 pm on September 4, 2024:

    Followup points:

    • I misunderstood “the build system may assign” to mean “cmake may assign” but actually it is referring to our own ProcessConfigurations.cmake file. So I think a more accurate comment than what I suggested would be “Set CMAKE_{C,CXX}_FLAGS_DEBUG to empty so the default options normally used for debug builds do not override MSAN options in CMAKE_{C,CXX}_FLAGS”

    • I’m probably missing something, but maybe a cleaner workaround instead of setting Debug build type and then erasing all the debug flags is just to set a custom build type -DCMAKE_BUILD_TYPE=msan that doesn’t have the flags to begin with?

    • Probably not applicable to this case, but it does seem unfortunate that only having FLAGS and FLAGS_<build_type> options forces a choice between specifying lower priority flags independent of build type, or higher priority flags tied to the build type. Ideally there would be a third option to specify high priority flags not depending on build type, maybe called FLAGS_ALL that would be applied to all build configurations. This seems like something that would not be too hard to support. Not important here, but maybe something to look into if there are more cases like this.


    hebasto commented at 11:41 am on September 6, 2024:
    • I’m probably missing something, but maybe a cleaner workaround instead of setting Debug build type and then erasing all the debug flags is just to set a custom build type -DCMAKE_BUILD_TYPE=msan that doesn’t have the flags to begin with?

    I agree with you, and that was my initial preference when I worked on the CMake staging branch. However, using undefined build configurations is not documented. As a result, the rough consensus among the CMake staging branch contributors was to adhere to documented practices.

    • Probably not applicable to this case, but it does seem unfortunate that only having FLAGS and FLAGS_<build_type> options forces a choice between specifying lower priority flags independent of build type, or higher priority flags tied to the build type. Ideally there would be a third option to specify high priority flags not depending on build type, maybe called FLAGS_ALL that would be applied to all build configurations. This seems like something that would not be too hard to support. Not important here, but maybe something to look into if there are more cases like this.

    Such an option is implemented in the APPEND_* variables, if I understood you correctly:https://github.com/bitcoin/bitcoin/blob/0e5cd608da5d8c3d9c758dbfa3fc36df4af4a100/CMakeLists.txt#L203-L206


github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin/bitcoin. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2024-09-20 01:12 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me