]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/llvm/tools/clang/lib/ARCMigrate/PlistReporter.cpp
Merge clang 7.0.1 and several follow-up changes
[FreeBSD/FreeBSD.git] / contrib / llvm / tools / clang / lib / ARCMigrate / PlistReporter.cpp
1 //===--- PlistReporter.cpp - ARC Migrate Tool Plist Reporter ----*- 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 #include "Internals.h"
11 #include "clang/Basic/FileManager.h"
12 #include "clang/Basic/PlistSupport.h"
13 #include "clang/Basic/SourceManager.h"
14 #include "clang/Lex/Lexer.h"
15 using namespace clang;
16 using namespace arcmt;
17 using namespace markup;
18
19 static StringRef getLevelName(DiagnosticsEngine::Level Level) {
20   switch (Level) {
21   case DiagnosticsEngine::Ignored:
22     llvm_unreachable("ignored");
23   case DiagnosticsEngine::Note:
24     return "note";
25   case DiagnosticsEngine::Remark:
26   case DiagnosticsEngine::Warning:
27     return "warning";
28   case DiagnosticsEngine::Fatal:
29   case DiagnosticsEngine::Error:
30     return "error";
31   }
32   llvm_unreachable("Invalid DiagnosticsEngine level!");
33 }
34
35 void arcmt::writeARCDiagsToPlist(const std::string &outPath,
36                                  ArrayRef<StoredDiagnostic> diags,
37                                  SourceManager &SM,
38                                  const LangOptions &LangOpts) {
39   DiagnosticIDs DiagIDs;
40
41   // Build up a set of FIDs that we use by scanning the locations and
42   // ranges of the diagnostics.
43   FIDMap FM;
44   SmallVector<FileID, 10> Fids;
45
46   for (ArrayRef<StoredDiagnostic>::iterator
47          I = diags.begin(), E = diags.end(); I != E; ++I) {
48     const StoredDiagnostic &D = *I;
49
50     AddFID(FM, Fids, SM, D.getLocation());
51
52     for (StoredDiagnostic::range_iterator
53            RI = D.range_begin(), RE = D.range_end(); RI != RE; ++RI) {
54       AddFID(FM, Fids, SM, RI->getBegin());
55       AddFID(FM, Fids, SM, RI->getEnd());
56     }
57   }
58
59   std::error_code EC;
60   llvm::raw_fd_ostream o(outPath, EC, llvm::sys::fs::F_Text);
61   if (EC) {
62     llvm::errs() << "error: could not create file: " << outPath << '\n';
63     return;
64   }
65
66   EmitPlistHeader(o);
67
68   // Write the root object: a <dict> containing...
69   //  - "files", an <array> mapping from FIDs to file names
70   //  - "diagnostics", an <array> containing the diagnostics
71   o << "<dict>\n"
72        " <key>files</key>\n"
73        " <array>\n";
74
75   for (FileID FID : Fids)
76     EmitString(o << "  ", SM.getFileEntryForID(FID)->getName()) << '\n';
77
78   o << " </array>\n"
79        " <key>diagnostics</key>\n"
80        " <array>\n";
81
82   for (ArrayRef<StoredDiagnostic>::iterator
83          DI = diags.begin(), DE = diags.end(); DI != DE; ++DI) {
84
85     const StoredDiagnostic &D = *DI;
86
87     if (D.getLevel() == DiagnosticsEngine::Ignored)
88       continue;
89
90     o << "  <dict>\n";
91
92     // Output the diagnostic.
93     o << "   <key>description</key>";
94     EmitString(o, D.getMessage()) << '\n';
95     o << "   <key>category</key>";
96     EmitString(o, DiagIDs.getCategoryNameFromID(
97                           DiagIDs.getCategoryNumberForDiag(D.getID()))) << '\n';
98     o << "   <key>type</key>";
99     EmitString(o, getLevelName(D.getLevel())) << '\n';
100
101     // Output the location of the bug.
102     o << "  <key>location</key>\n";
103     EmitLocation(o, SM, D.getLocation(), FM, 2);
104
105     // Output the ranges (if any).
106     if (!D.getRanges().empty()) {
107       o << "   <key>ranges</key>\n";
108       o << "   <array>\n";
109       for (auto &R : D.getRanges()) {
110         CharSourceRange ExpansionRange = SM.getExpansionRange(R);
111         EmitRange(o, SM, Lexer::getAsCharRange(ExpansionRange, SM, LangOpts),
112                   FM, 4);
113       }
114       o << "   </array>\n";
115     }
116
117     // Close up the entry.
118     o << "  </dict>\n";
119   }
120
121   o << " </array>\n";
122
123   // Finish.
124   o << "</dict>\n</plist>";
125 }