| //===- DiagnosticInfo.cpp -------------------------------------------------===// |
| // |
| // The MCLinker Project |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| #include <llvm/ADT/StringRef.h> |
| #include <llvm/Support/DataTypes.h> |
| |
| #include <mcld/ADT/SizeTraits.h> |
| #include <mcld/LinkerConfig.h> |
| #include <mcld/LD/Diagnostic.h> |
| #include <mcld/LD/DiagnosticInfos.h> |
| #include <mcld/LD/DiagnosticPrinter.h> |
| |
| #include <algorithm> |
| |
| using 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); } |
| }; |
| |
| } // namespace anonymous |
| |
| static const DiagStaticInfo DiagCommonInfo[] = { |
| #define DIAG(ENUM, CLASS, ADDRDESC, LOCDESC) \ |
| { diag::ENUM, CLASS, STR_SIZE(ADDRDESC, uint16_t), ADDRDESC }, |
| #include "mcld/LD/DiagCommonKinds.inc" |
| #include "mcld/LD/DiagReaders.inc" |
| #include "mcld/LD/DiagSymbolResolutions.inc" |
| #include "mcld/LD/DiagRelocations.inc" |
| #include "mcld/LD/DiagLayouts.inc" |
| #include "mcld/LD/DiagGOTPLT.inc" |
| #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/DiagReaders.inc" |
| #include "mcld/LD/DiagSymbolResolutions.inc" |
| #include "mcld/LD/DiagRelocations.inc" |
| #include "mcld/LD/DiagLayouts.inc" |
| #include "mcld/LD/DiagGOTPLT.inc" |
| #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().hasMulDefs()) { |
| severity = DiagnosticEngine::Ignore; |
| } |
| break; |
| } |
| case diag::undefined_reference: { |
| // 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() || !m_Config.options().isAllowShlibUndefined()) |
| severity = DiagnosticEngine::Error; |
| else |
| severity = DiagnosticEngine::Ignore; |
| break; |
| case LinkerConfig::Exec: |
| default: |
| severity = DiagnosticEngine::Error; |
| break; |
| } |
| 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; |
| } |
| |