blob: 468b6c89f2bc5f76475a36e979786d8d14a4da9e [file] [log] [blame]
Ted Kremenekd3abcdf2008-03-27 03:49:32 +00001//===--- PathDiagnostic.cpp - Path-Specific Diagnostic Handling -*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file defines the PathDiagnostic-related interfaces.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/Analysis/PathDiagnostic.h"
Chris Lattnerf4c83962008-11-19 06:51:40 +000015#include "llvm/ADT/SmallString.h"
Ted Kremenek120187d2008-03-27 06:16:40 +000016#include <sstream>
Ted Kremenekd3abcdf2008-03-27 03:49:32 +000017using namespace clang;
Ted Kremenek8af29752009-02-26 21:30:32 +000018
19static size_t GetNumCharsToLastNonPeriod(const char *s) {
20 const char *start = s;
21 const char *lastNonPeriod = 0;
22
23 for ( ; *s != '\0' ; ++s)
24 if (*s != '.') lastNonPeriod = s;
Ted Kremenekd3abcdf2008-03-27 03:49:32 +000025
Ted Kremenek8af29752009-02-26 21:30:32 +000026 if (!lastNonPeriod)
27 return 0;
28
29 return (lastNonPeriod - start) + 1;
30}
31
32static inline size_t GetNumCharsToLastNonPeriod(const std::string &s) {
33 return s.empty () ? 0 : GetNumCharsToLastNonPeriod(&s[0]);
34}
35
36PathDiagnosticPiece::PathDiagnosticPiece(FullSourceLoc pos,
37 const std::string& s,
Ted Kremeneke3ce2652009-03-02 19:39:50 +000038 Kind k, DisplayHint hint)
Ted Kremenek48504512009-03-06 07:53:30 +000039 : Pos(pos), str(s, 0, GetNumCharsToLastNonPeriod(s)), kind(k), Hint(hint) {
40 assert(Pos.isValid() &&
41 "PathDiagnosticPiece's must have a valid location.");
42}
Ted Kremenek8af29752009-02-26 21:30:32 +000043
44PathDiagnosticPiece::PathDiagnosticPiece(FullSourceLoc pos,
Ted Kremeneke3ce2652009-03-02 19:39:50 +000045 const char* s, Kind k,
Ted Kremenek8af29752009-02-26 21:30:32 +000046 DisplayHint hint)
Ted Kremenek48504512009-03-06 07:53:30 +000047 : Pos(pos), str(s, GetNumCharsToLastNonPeriod(s)), kind(k), Hint(hint) {
48 assert(Pos.isValid() &&
49 "PathDiagnosticPiece's must have a valid location.");
50}
51
52PathDiagnostic::PathDiagnostic() : Size(0) {}
Ted Kremenek8af29752009-02-26 21:30:32 +000053
Ted Kremenekd3abcdf2008-03-27 03:49:32 +000054PathDiagnostic::~PathDiagnostic() {
55 for (iterator I = begin(), E = end(); I != E; ++I) delete &*I;
56}
57
Ted Kremeneka127cca2009-03-06 07:08:50 +000058
59PathDiagnostic::PathDiagnostic(const char* bugtype, const char* desc,
60 const char* category)
Ted Kremenek48504512009-03-06 07:53:30 +000061 : Size(0),
62 BugType(bugtype, GetNumCharsToLastNonPeriod(bugtype)),
Ted Kremeneka127cca2009-03-06 07:08:50 +000063 Desc(desc, GetNumCharsToLastNonPeriod(desc)),
64 Category(category, GetNumCharsToLastNonPeriod(category)) {}
65
66PathDiagnostic::PathDiagnostic(const std::string& bugtype,
67 const std::string& desc,
68 const std::string& category)
Ted Kremenek48504512009-03-06 07:53:30 +000069 : Size(0),
70 BugType(bugtype, 0, GetNumCharsToLastNonPeriod(bugtype)),
Ted Kremeneka127cca2009-03-06 07:08:50 +000071 Desc(desc, 0, GetNumCharsToLastNonPeriod(desc)),
72 Category(category, 0, GetNumCharsToLastNonPeriod(category)) {}
73
Chris Lattner0a14eee2008-11-18 07:04:44 +000074void PathDiagnosticClient::HandleDiagnostic(Diagnostic::Level DiagLevel,
75 const DiagnosticInfo &Info) {
Ted Kremenekd3abcdf2008-03-27 03:49:32 +000076
77 // Create a PathDiagnostic with a single piece.
78
Ted Kremenek55851142008-04-22 16:15:03 +000079 PathDiagnostic* D = new PathDiagnostic();
Ted Kremenekd3abcdf2008-03-27 03:49:32 +000080
Chris Lattnere837f932008-11-18 04:44:58 +000081 const char *LevelStr;
Ted Kremenek120187d2008-03-27 06:16:40 +000082 switch (DiagLevel) {
Mike Stumpe87f5c12009-02-07 03:46:08 +000083 default:
Chris Lattner41327582009-02-06 03:57:44 +000084 case Diagnostic::Ignored: assert(0 && "Invalid diagnostic type");
Chris Lattnere837f932008-11-18 04:44:58 +000085 case Diagnostic::Note: LevelStr = "note: "; break;
86 case Diagnostic::Warning: LevelStr = "warning: "; break;
87 case Diagnostic::Error: LevelStr = "error: "; break;
Chris Lattner41327582009-02-06 03:57:44 +000088 case Diagnostic::Fatal: LevelStr = "fatal error: "; break;
Ted Kremenekd3abcdf2008-03-27 03:49:32 +000089 }
Ted Kremenek120187d2008-03-27 06:16:40 +000090
Chris Lattnerf4c83962008-11-19 06:51:40 +000091 llvm::SmallString<100> StrC;
92 StrC += LevelStr;
93 Info.FormatDiagnostic(StrC);
Ted Kremenek120187d2008-03-27 06:16:40 +000094
Chris Lattner0a14eee2008-11-18 07:04:44 +000095 PathDiagnosticPiece *P =
Chris Lattnerf4c83962008-11-19 06:51:40 +000096 new PathDiagnosticPiece(Info.getLocation(),
97 std::string(StrC.begin(), StrC.end()));
Ted Kremenek120187d2008-03-27 06:16:40 +000098
Chris Lattner0a14eee2008-11-18 07:04:44 +000099 for (unsigned i = 0, e = Info.getNumRanges(); i != e; ++i)
100 P->addRange(Info.getRange(i));
Douglas Gregor4b2d3f72009-02-26 21:00:50 +0000101 for (unsigned i = 0, e = Info.getNumCodeModificationHints(); i != e; ++i)
102 P->addCodeModificationHint(Info.getCodeModificationHint(i));
Ted Kremenek55851142008-04-22 16:15:03 +0000103 D->push_front(P);
Ted Kremenekd3abcdf2008-03-27 03:49:32 +0000104
Ted Kremenek120187d2008-03-27 06:16:40 +0000105 HandlePathDiagnostic(D);
Ted Kremenekd3abcdf2008-03-27 03:49:32 +0000106}