Home | History | Annotate | Download | only in ARCMigrate
      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.getAsRange()),
    111                                        R.isTokenRange());
    112         EmitRange(o, SM, Lexer::getAsCharRange(ExpansionRange, SM, LangOpts),
    113                   FM, 4);
    114       }
    115       o << "   </array>\n";
    116     }
    117 
    118     // Close up the entry.
    119     o << "  </dict>\n";
    120   }
    121 
    122   o << " </array>\n";
    123 
    124   // Finish.
    125   o << "</dict>\n</plist>";
    126 }
    127