//===- DiagnosticInfo.cpp -------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include "mcld/LD/DiagnosticInfos.h"
#include "mcld/LinkerConfig.h"
#include "mcld/ADT/SizeTraits.h"
#include "mcld/LD/Diagnostic.h"
#include "mcld/LD/DiagnosticPrinter.h"
#include <llvm/ADT/StringRef.h>
#include <llvm/Support/DataTypes.h>
#include <algorithm>
namespace mcld {
namespace {
struct DiagStaticInfo {
public:
uint16_t ID;
DiagnosticEngine::Severity Severity;
uint16_t DescriptionLen;
const char* DescriptionStr;
public:
llvm::StringRef getDescription() const {
return llvm::StringRef(DescriptionStr, DescriptionLen);
}
bool operator<(const DiagStaticInfo& pRHS) const { return (ID < pRHS.ID); }
};
} // anonymous namespace
static const DiagStaticInfo DiagCommonInfo[] = {
#define DIAG(ENUM, CLASS, ADDRDESC, LOCDESC) \
{ diag::ENUM, CLASS, STR_SIZE(ADDRDESC, uint16_t), ADDRDESC } \
,
#include "mcld/LD/DiagAttribute.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagCommonKinds.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagReaders.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagSymbolResolutions.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagRelocations.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagLayouts.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagGOTPLT.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagLDScript.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagMips.inc" // NOLINT [build/include] [4]
#undef DIAG
{0, DiagnosticEngine::None, 0, 0}};
static const unsigned int DiagCommonInfoSize =
sizeof(DiagCommonInfo) / sizeof(DiagCommonInfo[0]) - 1;
static const DiagStaticInfo DiagLoCInfo[] = {
#define DIAG(ENUM, CLASS, ADDRDESC, LOCDESC) \
{ diag::ENUM, CLASS, STR_SIZE(LOCDESC, uint16_t), LOCDESC } \
,
#include "mcld/LD/DiagAttribute.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagCommonKinds.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagReaders.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagSymbolResolutions.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagRelocations.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagLayouts.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagGOTPLT.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagLDScript.inc" // NOLINT [build/include] [4]
#include "mcld/LD/DiagMips.inc" // NOLINT [build/include] [4]
#undef DIAG
{0, DiagnosticEngine::None, 0, 0}};
static const unsigned int DiagLoCInfoSize =
sizeof(DiagLoCInfo) / sizeof(DiagLoCInfo[0]) - 1;
static const DiagStaticInfo* getDiagInfo(unsigned int pID,
bool pInLoC = false) {
const DiagStaticInfo* static_info = (pInLoC) ? DiagLoCInfo : DiagCommonInfo;
unsigned int info_size = (pInLoC) ? DiagLoCInfoSize : DiagCommonInfoSize;
DiagStaticInfo key = {
static_cast<uint16_t>(pID), DiagnosticEngine::None, 0, 0};
const DiagStaticInfo* result =
std::lower_bound(static_info, static_info + info_size, key);
if (result == (static_info + info_size) || result->ID != pID)
return NULL;
return result;
}
//===----------------------------------------------------------------------===//
// DiagnosticInfos
//===----------------------------------------------------------------------===//
DiagnosticInfos::DiagnosticInfos(const LinkerConfig& pConfig)
: m_Config(pConfig) {
}
DiagnosticInfos::~DiagnosticInfos() {
}
llvm::StringRef DiagnosticInfos::getDescription(unsigned int pID,
bool pInLoC) const {
return getDiagInfo(pID, pInLoC)->getDescription();
}
bool DiagnosticInfos::process(DiagnosticEngine& pEngine) const {
Diagnostic info(pEngine);
unsigned int ID = info.getID();
// we are not implement LineInfo, so keep pIsLoC false.
const DiagStaticInfo* static_info = getDiagInfo(ID);
DiagnosticEngine::Severity severity = static_info->Severity;
switch (ID) {
case diag::multiple_definitions: {
if (m_Config.options().isMulDefs()) {
severity = DiagnosticEngine::Ignore;
}
break;
}
case diag::undefined_reference:
case diag::undefined_reference_text: {
// we have not implement --unresolved-symbols=method yet. So far, MCLinker
// provides the easier --allow-shlib-undefined and --no-undefined (i.e.
// -z defs)
switch (m_Config.codeGenType()) {
case LinkerConfig::Object:
if (m_Config.options().isNoUndefined())
severity = DiagnosticEngine::Error;
else
severity = DiagnosticEngine::Ignore;
break;
case LinkerConfig::DynObj:
if (m_Config.options().isNoUndefined())
severity = DiagnosticEngine::Error;
else
severity = DiagnosticEngine::Ignore;
break;
default:
severity = DiagnosticEngine::Error;
break;
}
break;
}
case diag::debug_print_gc_sections: {
if (!m_Config.options().getPrintGCSections())
severity = DiagnosticEngine::Ignore;
break;
}
default:
break;
} // end of switch
// If --fatal-warnings is turned on, then switch warnings and errors to fatal
if (m_Config.options().isFatalWarnings()) {
if (severity == DiagnosticEngine::Warning ||
severity == DiagnosticEngine::Error) {
severity = DiagnosticEngine::Fatal;
}
}
// finally, report it.
pEngine.getPrinter()->handleDiagnostic(severity, info);
return true;
}
} // namespace mcld