From 9ee1b6137acc6d74dba6d27aafd2395b7ec87075 Mon Sep 17 00:00:00 2001 From: Nerixyz Date: Wed, 21 May 2025 17:32:48 +0200 Subject: [PATCH] [LLDB] Avoid crashes when inspecting MS STL types --- .../Language/CPlusPlus/GenericOptional.cpp | 7 ++- .../Plugins/Language/CPlusPlus/LibStdcpp.cpp | 3 + .../Shell/Process/Windows/msstl_smoke.cpp | 56 +++++++++++++++++++ 3 files changed, 63 insertions(+), 3 deletions(-) create mode 100644 lldb/test/Shell/Process/Windows/msstl_smoke.cpp diff --git a/lldb/source/Plugins/Language/CPlusPlus/GenericOptional.cpp b/lldb/source/Plugins/Language/CPlusPlus/GenericOptional.cpp index b1fdc0fe37763..c041f39022d10 100644 --- a/lldb/source/Plugins/Language/CPlusPlus/GenericOptional.cpp +++ b/lldb/source/Plugins/Language/CPlusPlus/GenericOptional.cpp @@ -74,9 +74,10 @@ lldb::ChildCacheState GenericOptionalFrontend::Update() { if (m_stdlib == StdLib::LibCxx) engaged_sp = m_backend.GetChildMemberWithName("__engaged_"); - else if (m_stdlib == StdLib::LibStdcpp) - engaged_sp = m_backend.GetChildMemberWithName("_M_payload") - ->GetChildMemberWithName("_M_engaged"); + else if (m_stdlib == StdLib::LibStdcpp) { + if (ValueObjectSP payload = m_backend.GetChildMemberWithName("_M_payload")) + engaged_sp = payload->GetChildMemberWithName("_M_engaged"); + } if (!engaged_sp) return lldb::ChildCacheState::eRefetch; diff --git a/lldb/source/Plugins/Language/CPlusPlus/LibStdcpp.cpp b/lldb/source/Plugins/Language/CPlusPlus/LibStdcpp.cpp index 02113baf64b8c..08cfcd4a26b8e 100644 --- a/lldb/source/Plugins/Language/CPlusPlus/LibStdcpp.cpp +++ b/lldb/source/Plugins/Language/CPlusPlus/LibStdcpp.cpp @@ -379,6 +379,9 @@ LibStdcppSharedPtrSyntheticFrontEnd::CalculateNumChildren() { lldb::ValueObjectSP LibStdcppSharedPtrSyntheticFrontEnd::GetChildAtIndex(uint32_t idx) { + if (!m_ptr_obj) + return nullptr; + if (idx == 0) return m_ptr_obj->GetSP(); if (idx == 1) { diff --git a/lldb/test/Shell/Process/Windows/msstl_smoke.cpp b/lldb/test/Shell/Process/Windows/msstl_smoke.cpp new file mode 100644 index 0000000000000..1dcb9c0093b10 --- /dev/null +++ b/lldb/test/Shell/Process/Windows/msstl_smoke.cpp @@ -0,0 +1,56 @@ +// This smoke test ensures that LLDB doesn't crash when formatting types from MSVC's STL. +// FIXME: LLDB currently has no built-in formatters for MSVC's STL (#24834) + +// REQUIRES: target-windows +// RUN: %build --compiler=clang-cl -o %t.exe --std c++20 -- %s +// RUN: env LLDB_USE_NATIVE_PDB_READER=1 %lldb -f %t.exe -o "b main" -o "run" -o "fr v" -o c | FileCheck %s + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +int main() { + std::shared_ptr foo; + std::weak_ptr weak = foo; + std::unique_ptr unique(new int(42)); + std::optional opt; + std::string str = "str"; + std::string longStr = "string that is long enough such that no SSO can happen"; + std::wstring wStr = L"wstr"; + std::wstring longWStr = L"string that is long enough such that no SSO can happen"; + std::tuple tuple{1, false, 4.2}; + std::coroutine_handle<> coroHandle; + std::bitset<16> bitset(123); + + std::map map{{1, 2}, {2, 4}, {3, 6}, {4, 8}, {5, 10}}; + auto mapIt = map.find(3); + auto mapItEnd = map.find(9); + std::set set{1, 2, 3}; + std::multimap mMap{{1, 2}, {1, 1}, {2, 4}}; + std::multiset mSet{1, 2, 3}; + + std::variant variant; + std::list list{1, 2, 3}; + std::forward_list fwList{1, 2, 3}; + + std::unordered_map uMap{{1, 2}, {2, 4}, {3, 6}}; + std::unordered_set uSet{1, 2, 4}; + std::unordered_multimap uMMap{{1, 2}, {1, 1}, {2, 4}}; + std::unordered_multiset uMSet{1, 1, 2}; + std::deque deque{1, 2, 3}; + std::vector vec{1, 2, 3}; +} + +// CHECK: Process {{.*}} exited with status = 0 (0x00000000)