Index: ELF/Config.h =================================================================== --- ELF/Config.h +++ ELF/Config.h @@ -144,6 +144,7 @@ ELFKind EKind = ELFNoneKind; uint16_t DefaultSymbolVersion = llvm::ELF::VER_NDX_GLOBAL; uint16_t EMachine = llvm::ELF::EM_NONE; + uint32_t ErrorLimit; uint64_t EntryAddr = 0; uint64_t ImageBase; uint64_t MaxPageSize; Index: ELF/Driver.cpp =================================================================== --- ELF/Driver.cpp +++ ELF/Driver.cpp @@ -43,6 +43,7 @@ bool elf::link(ArrayRef Args, bool CanExitEarly, raw_ostream &Error) { HasError = false; + ErrorCount = 0; ErrorOS = &Error; Argv0 = Args[0]; @@ -490,6 +491,7 @@ Config->Discard = getDiscardOption(Args); Config->EhFrameHdr = Args.hasArg(OPT_eh_frame_hdr); Config->EnableNewDtags = !Args.hasArg(OPT_disable_new_dtags); + Config->ErrorLimit = getInteger(Args, OPT_error_limit, 20); Config->ExportDynamic = Args.hasArg(OPT_export_dynamic); Config->FatalWarnings = Args.hasArg(OPT_fatal_warnings); Config->GcSections = getArg(Args, OPT_gc_sections, OPT_no_gc_sections, false); Index: ELF/Error.h =================================================================== --- ELF/Error.h +++ ELF/Error.h @@ -32,6 +32,7 @@ namespace elf { extern bool HasError; +extern unsigned ErrorCount; extern llvm::raw_ostream *ErrorOS; extern llvm::StringRef Argv0; Index: ELF/Error.cpp =================================================================== --- ELF/Error.cpp +++ ELF/Error.cpp @@ -24,6 +24,7 @@ namespace lld { bool elf::HasError; +unsigned elf::ErrorCount; raw_ostream *elf::ErrorOS; StringRef elf::Argv0; @@ -40,8 +41,16 @@ } void elf::error(const Twine &Msg) { - *ErrorOS << Argv0 << ": error: " << Msg << "\n"; + if (Config->ErrorLimit == 0 || ErrorCount < Config->ErrorLimit) { + *ErrorOS << Argv0 << ": error: " << Msg << "\n"; + } else if (ErrorCount == Config->ErrorLimit) { + *ErrorOS << Argv0 << ": error: too many errors emitted, stopping now\n"; + if (Config->ExitEarly) + exitLld(1); + } + HasError = true; + ErrorCount++; } void elf::error(std::error_code EC, const Twine &Prefix) { Index: ELF/Options.td =================================================================== --- ELF/Options.td +++ ELF/Options.td @@ -68,6 +68,9 @@ def entry: S<"entry">, MetaVarName<"">, HelpText<"Name of entry point symbol">; +def error_limit: S<"error-limit">, + HelpText<"Maximum number of errors to emit before stopping (0 = no limit)">; + def export_dynamic: F<"export-dynamic">, HelpText<"Put symbols in the dynamic symbol table">; @@ -240,6 +243,7 @@ def alias_dynamic_list: J<"dynamic-list=">, Alias; def alias_entry_e: JoinedOrSeparate<["-"], "e">, Alias; def alias_entry_entry: J<"entry=">, Alias; +def alias_error_limit: J<"error-limit=">, Alias; def alias_export_dynamic_E: Flag<["-"], "E">, Alias; def alias_export_dynamic_symbol: J<"export-dynamic-symbol=">, Alias; Index: test/ELF/error-limit.test =================================================================== --- /dev/null +++ test/ELF/error-limit.test @@ -0,0 +1,24 @@ +RUN: not ld.lld 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 2>&1 | \ +RUN: FileCheck -check-prefix=DEFAULT %s + +DEFAULT: cannot open 01 +DEFAULT: cannot open 20 +DEFAULT-NEXT: too many errors emitted, stopping now +DEFAULT-NOT: cannot open 21 + +RUN: not ld.lld -error-limit=5 01 02 03 04 05 06 07 08 09 10 2>&1 | FileCheck -check-prefix=LIMIT5 %s +RUN: not ld.lld -error-limit 5 01 02 03 04 05 06 07 08 09 10 2>&1 | FileCheck -check-prefix=LIMIT5 %s + +LIMIT5: cannot open 01 +LIMIT5: cannot open 05 +LIMIT5-NEXT: too many errors emitted, stopping now +LIMIT5-NOT: cannot open 06 + +RUN: not ld.lld -error-limit=0 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 2>&1 | \ +RUN: FileCheck -check-prefix=UNLIMITED %s + +UNLIMITED: cannot open 01 +UNLIMITED: cannot open 20 +UNLIMITED: cannot open 21 +UNLIMITED: cannot open 22 +UNLIMITED-NOT: too many errors emitted, stopping now