This is a follow-up for D70378 (Cover usage of LLD as a library). While debugging an intermittent failure on a bot, I recalled this scenario which causes the issue: 1.When executing lld/test/ELF/invalid/symtab-sh-info.s L45, we reach lld::elf::Obj-File::ObjFile() which goes straight into its base ELFFileBase(), then ELFFileBase::init(). 2.At that point fatal() is thrown in lld/ELF/InputFiles.cpp L381, leaving a half-initialized ObjFile instance. 3.We then end up in lld::exitLld() and since we are running with LLD_IN_TEST, we hapily restore the control flow to CrashRecoveryContext::RunSafely() then back in lld::safeLldMain(). 4.Before this patch, we called errorHandler().reset() just after, and this attempted to reset the associated SpecificAlloc<ObjFile<ELF64LE>>. That tried to free the half-initialized ObjFile instance, and more precisely its ObjFile::dwarf member. Sometimes that worked, sometimes it failed and was catched by the CrashRecoveryContext. This scenario was the reason we called errorHandler().reset() through a CrashRecoveryContext. But in some rare cases, the above repro somehow corrupted the heap, creating a stack overflow. When the CrashRecoveryContext's filter (that is, __except (ExceptionFilter(GetExceptionInformation()))) tried to handle the exception, it crashed again since the stack was exhausted -- and that took the whole application down. That is the issue seen on the bot. Locally it happens about 1 times out of 15. Now this situation can happen anywhere in LLD. Since catching stack overflows is not a reliable scenario ATM when using CrashRecoveryContext, we're now preventing further re-entrance when such failures occur, by signaling lld::SafeReturn::canRunAgain=false. When running with LLD_IN_TEST=2 (or above), only one iteration will be executed, instead of two. Differential Revision: https://reviews.llvm.org/D88348
62 lines
2.0 KiB
C++
62 lines
2.0 KiB
C++
//===- lld/Common/Driver.h - Linker Driver Emulator -----------------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLD_COMMON_DRIVER_H
|
|
#define LLD_COMMON_DRIVER_H
|
|
|
|
#include "llvm/ADT/ArrayRef.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
namespace lld {
|
|
struct SafeReturn {
|
|
int ret;
|
|
bool canRunAgain;
|
|
};
|
|
|
|
// Generic entry point when using LLD as a library, safe for re-entry, supports
|
|
// crash recovery. Returns a general completion code and a boolean telling
|
|
// whether it can be called again. In some cases, a crash could corrupt memory
|
|
// and re-entry would not be possible anymore. Use exitLld() in that case to
|
|
// properly exit your application and avoid intermittent crashes on exit caused
|
|
// by cleanup.
|
|
SafeReturn safeLldMain(int argc, const char **argv, llvm::raw_ostream &stdoutOS,
|
|
llvm::raw_ostream &stderrOS);
|
|
|
|
namespace coff {
|
|
bool link(llvm::ArrayRef<const char *> args, bool canExitEarly,
|
|
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS);
|
|
}
|
|
|
|
namespace mingw {
|
|
bool link(llvm::ArrayRef<const char *> args, bool canExitEarly,
|
|
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS);
|
|
}
|
|
|
|
namespace elf {
|
|
bool link(llvm::ArrayRef<const char *> args, bool canExitEarly,
|
|
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS);
|
|
}
|
|
|
|
namespace mach_o {
|
|
bool link(llvm::ArrayRef<const char *> args, bool canExitEarly,
|
|
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS);
|
|
}
|
|
|
|
namespace macho {
|
|
bool link(llvm::ArrayRef<const char *> args, bool canExitEarly,
|
|
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS);
|
|
}
|
|
|
|
namespace wasm {
|
|
bool link(llvm::ArrayRef<const char *> args, bool canExitEarly,
|
|
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS);
|
|
}
|
|
}
|
|
|
|
#endif
|