blob: 9a731587e5e0c6a4cc4e00155877a707d793084f [file] [log] [blame]
//===- DiagnosticNames.h - Defines a table of all builtin diagnostics ------==//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/StringRef.h"
#include "llvm/Support/DataTypes.h"
namespace diagtool {
struct DiagnosticRecord {
const char *NameStr;
short DiagID;
uint8_t NameLen;
llvm::StringRef getName() const {
return llvm::StringRef(NameStr, NameLen);
}
bool operator<(const DiagnosticRecord &Other) const {
return getName() < Other.getName();
}
};
/// \brief Get every diagnostic in the system, sorted by name.
llvm::ArrayRef<DiagnosticRecord> getBuiltinDiagnosticsByName();
/// \brief Get a diagnostic by its ID.
const DiagnosticRecord &getDiagnosticForID(short DiagID);
struct GroupRecord {
// Be safe with the size of 'NameLen' because we don't statically check if
// the size will fit in the field; the struct size won't decrease with a
// shorter type anyway.
size_t NameLen;
const char *NameStr;
const short *Members;
const short *SubGroups;
llvm::StringRef getName() const {
return llvm::StringRef(NameStr, NameLen);
}
template<typename RecordType>
class group_iterator {
const short *CurrentID;
friend struct GroupRecord;
group_iterator(const short *Start) : CurrentID(Start) {
if (CurrentID && *CurrentID == -1)
CurrentID = 0;
}
public:
typedef RecordType value_type;
typedef const value_type & reference;
typedef const value_type * pointer;
typedef std::forward_iterator_tag iterator_category;
typedef std::ptrdiff_t difference_type;
inline reference operator*() const;
inline pointer operator->() const {
return &operator*();
}
inline short getID() const {
return *CurrentID;
}
group_iterator &operator++() {
++CurrentID;
if (*CurrentID == -1)
CurrentID = 0;
return *this;
}
bool operator==(group_iterator &Other) const {
return CurrentID == Other.CurrentID;
}
bool operator!=(group_iterator &Other) const {
return CurrentID != Other.CurrentID;
}
};
typedef group_iterator<GroupRecord> subgroup_iterator;
subgroup_iterator subgroup_begin() const {
return SubGroups;
}
subgroup_iterator subgroup_end() const {
return 0;
}
typedef group_iterator<DiagnosticRecord> diagnostics_iterator;
diagnostics_iterator diagnostics_begin() const {
return Members;
}
diagnostics_iterator diagnostics_end() const {
return 0;
}
bool operator<(const GroupRecord &Other) const {
return getName() < Other.getName();
}
};
/// \brief Get every diagnostic group in the system, sorted by name.
llvm::ArrayRef<GroupRecord> getDiagnosticGroups();
template<>
inline GroupRecord::subgroup_iterator::reference
GroupRecord::subgroup_iterator::operator*() const {
return getDiagnosticGroups()[*CurrentID];
}
template<>
inline GroupRecord::diagnostics_iterator::reference
GroupRecord::diagnostics_iterator::operator*() const {
return getDiagnosticForID(*CurrentID);
}
} // end namespace diagtool