Skip to content

Commit eda72fa

Browse files
authoredSep 18, 2024
Fix OOM in FormatDiagnostic (2nd attempt) (#108866)
Resolves: #70930 (and probably latest comments from clangd/clangd#251) by fixing racing for the shared DiagStorage value which caused messing with args inside the storage and then formatting the following message with getArgSInt(1) == 2: def err_module_odr_violation_function : Error< "%q0 has different definitions in different modules; " "%select{definition in module '%2'|defined here}1 " "first difference is " which causes HandleSelectModifier to go beyond the ArgumentLen so the recursive call to FormatDiagnostic was made with DiagStr > DiagEnd that leads to infinite while (DiagStr != DiagEnd). The Main Idea: Reuse the existing DiagStorageAllocator logic to make all DiagnosticBuilders having independent states. Also, encapsulating the rest of state (e.g. ID and Loc) into DiagnosticBuilder. The last attempt failed - #108187 (comment) so was reverted - #108838
1 parent 2c85fe9 commit eda72fa

File tree

18 files changed

+233
-307
lines changed

18 files changed

+233
-307
lines changed
 

‎clang-tools-extra/clang-tidy/ClangTidyDiagnosticConsumer.cpp

-2
Original file line numberDiff line numberDiff line change
@@ -380,7 +380,6 @@ void ClangTidyDiagnosticConsumer::HandleDiagnostic(
380380
++Context.Stats.ErrorsIgnoredNOLINT;
381381
// Ignored a warning, should ignore related notes as well
382382
LastErrorWasIgnored = true;
383-
Context.DiagEngine->Clear();
384383
for (const auto &Error : SuppressionErrors)
385384
Context.diag(Error);
386385
return;
@@ -457,7 +456,6 @@ void ClangTidyDiagnosticConsumer::HandleDiagnostic(
457456
if (Info.hasSourceManager())
458457
checkFilters(Info.getLocation(), Info.getSourceManager());
459458

460-
Context.DiagEngine->Clear();
461459
for (const auto &Error : SuppressionErrors)
462460
Context.diag(Error);
463461
}

‎clang-tools-extra/clangd/unittests/ConfigCompileTests.cpp

+6-6
Original file line numberDiff line numberDiff line change
@@ -305,33 +305,33 @@ TEST_F(ConfigCompileTests, DiagnosticSuppression) {
305305
{
306306
auto D = DiagEngine.Report(diag::warn_unreachable);
307307
EXPECT_TRUE(isDiagnosticSuppressed(
308-
Diag{&DiagEngine}, Conf.Diagnostics.Suppress, LangOptions()));
308+
Diag{&DiagEngine, D}, Conf.Diagnostics.Suppress, LangOptions()));
309309
}
310310
// Subcategory not respected/suppressed.
311311
{
312312
auto D = DiagEngine.Report(diag::warn_unreachable_break);
313313
EXPECT_FALSE(isDiagnosticSuppressed(
314-
Diag{&DiagEngine}, Conf.Diagnostics.Suppress, LangOptions()));
314+
Diag{&DiagEngine, D}, Conf.Diagnostics.Suppress, LangOptions()));
315315
}
316316
{
317317
auto D = DiagEngine.Report(diag::warn_unused_variable);
318318
EXPECT_TRUE(isDiagnosticSuppressed(
319-
Diag{&DiagEngine}, Conf.Diagnostics.Suppress, LangOptions()));
319+
Diag{&DiagEngine, D}, Conf.Diagnostics.Suppress, LangOptions()));
320320
}
321321
{
322322
auto D = DiagEngine.Report(diag::err_typecheck_bool_condition);
323323
EXPECT_TRUE(isDiagnosticSuppressed(
324-
Diag{&DiagEngine}, Conf.Diagnostics.Suppress, LangOptions()));
324+
Diag{&DiagEngine, D}, Conf.Diagnostics.Suppress, LangOptions()));
325325
}
326326
{
327327
auto D = DiagEngine.Report(diag::err_unexpected_friend);
328328
EXPECT_TRUE(isDiagnosticSuppressed(
329-
Diag{&DiagEngine}, Conf.Diagnostics.Suppress, LangOptions()));
329+
Diag{&DiagEngine, D}, Conf.Diagnostics.Suppress, LangOptions()));
330330
}
331331
{
332332
auto D = DiagEngine.Report(diag::warn_alloca);
333333
EXPECT_TRUE(isDiagnosticSuppressed(
334-
Diag{&DiagEngine}, Conf.Diagnostics.Suppress, LangOptions()));
334+
Diag{&DiagEngine, D}, Conf.Diagnostics.Suppress, LangOptions()));
335335
}
336336

337337
Frag.Diagnostics.Suppress.emplace_back("*");

0 commit comments

Comments
 (0)