Skip to content
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 16 additions & 6 deletions lldb/source/Target/Language.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -542,9 +542,22 @@ Language::Language() = default;
// Destructor
Language::~Language() = default;

static llvm::dwarf::SourceLanguage
ToDwarfSourceLanguage(lldb::LanguageType language_type) {
if (language_type < lldb::eLanguageTypeLastStandardLanguage)
return static_cast<llvm::dwarf::SourceLanguage>(language_type);

switch (language_type) {
case eLanguageTypeMipsAssembler:
return llvm::dwarf::DW_LANG_Mips_Assembler;
default: break;
}

llvm_unreachable("Unhandled language type");
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are we sure we cannot get here even if. we parse DWARF with e.g., a DW_LANG_user_low+n language attribute?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Otherwise a fail-safe that doesn't crash might be more appropriate.

Copy link
Member Author

@Michael137 Michael137 Oct 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

true if the debuggee was compiled with a compiler that emitted a vendor language that LLDB isn't aware of. That would hit this codepath. Let me relax this a bit

}

SourceLanguage::SourceLanguage(lldb::LanguageType language_type) {
auto lname =
llvm::dwarf::toDW_LNAME((llvm::dwarf::SourceLanguage)language_type);
auto lname = llvm::dwarf::toDW_LNAME(ToDwarfSourceLanguage(language_type));
if (!lname)
return;
name = lname->first;
Expand All @@ -559,11 +572,8 @@ lldb::LanguageType SourceLanguage::AsLanguageType() const {
}

llvm::StringRef SourceLanguage::GetDescription() const {
LanguageType type = AsLanguageType();
if (type)
return Language::GetNameForLanguageType(type);
return llvm::dwarf::LanguageDescription(
(llvm::dwarf::SourceLanguageName)name);
static_cast<llvm::dwarf::SourceLanguageName>(name));
}
bool SourceLanguage::IsC() const { return name == llvm::dwarf::DW_LNAME_C; }

Expand Down
1 change: 1 addition & 0 deletions lldb/unittests/Target/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ add_lldb_unittest(TargetTests
ABITest.cpp
DynamicRegisterInfoTest.cpp
ExecutionContextTest.cpp
Language.cpp
LocateModuleCallbackTest.cpp
MemoryRegionInfoTest.cpp
MemoryTest.cpp
Expand Down
69 changes: 69 additions & 0 deletions lldb/unittests/Target/Language.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
//===-- LanguageTest.cpp --------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

#include "lldb/Target/Language.h"
#include "lldb/lldb-enumerations.h"
#include "gtest/gtest.h"

using namespace lldb_private;
using namespace lldb;

namespace {
class LanguageTest : public ::testing::Test {};
} // namespace

TEST_F(LanguageTest, SourceLanguage_GetDescription) {
for (uint32_t i = 1; i < lldb::eNumLanguageTypes; ++i) {
// 0x29 is unassigned
if (i == 0x29)
continue;

auto lang_type = static_cast<lldb::LanguageType>(i);
if (lang_type == lldb::eLanguageTypeLastStandardLanguage)
continue;

SourceLanguage lang(lang_type);

// eLanguageTypeHIP is not implemented as a DW_LNAME because of a conflict.
if (lang_type == lldb::eLanguageTypeHIP)
EXPECT_FALSE(lang);
else
EXPECT_TRUE(lang);
}

EXPECT_EQ(SourceLanguage(eLanguageTypeC_plus_plus).GetDescription(),
"ISO C++");
EXPECT_EQ(SourceLanguage(eLanguageTypeC_plus_plus_17).GetDescription(),
"ISO C++");
EXPECT_EQ(SourceLanguage(eLanguageTypeC_plus_plus_20).GetDescription(),
"ISO C++");

EXPECT_EQ(SourceLanguage(eLanguageTypeObjC).GetDescription(), "Objective C");
EXPECT_EQ(SourceLanguage(eLanguageTypeMipsAssembler).GetDescription(),
"Assembly");

auto next_vendor_language =
static_cast<lldb::LanguageType>(eLanguageTypeMipsAssembler + 1);
if (next_vendor_language < eNumLanguageTypes)
EXPECT_NE(SourceLanguage(next_vendor_language).GetDescription(), "Unknown");

EXPECT_EQ(SourceLanguage(eLanguageTypeUnknown).GetDescription(), "Unknown");
}

TEST_F(LanguageTest, SourceLanguage_AsLanguageType) {
EXPECT_EQ(SourceLanguage(eLanguageTypeC_plus_plus).AsLanguageType(),
eLanguageTypeC_plus_plus);
EXPECT_EQ(SourceLanguage(eLanguageTypeC_plus_plus_03).AsLanguageType(),
eLanguageTypeC_plus_plus_03);

// Vendor-specific language code.
EXPECT_EQ(SourceLanguage(eLanguageTypeMipsAssembler).AsLanguageType(),
eLanguageTypeAssembly);
EXPECT_EQ(SourceLanguage(eLanguageTypeUnknown).AsLanguageType(),
eLanguageTypeUnknown);
}
Loading