xref: /openbsd-src/gnu/llvm/clang/include/clang/Basic/PlistSupport.h (revision e5dd70708596ae51455a0ffa086a00c5b29f8583)
1*e5dd7070Spatrick //===- PlistSupport.h - Plist Output Utilities ------------------*- C++ -*-===//
2*e5dd7070Spatrick //
3*e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5*e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*e5dd7070Spatrick //
7*e5dd7070Spatrick //===----------------------------------------------------------------------===//
8*e5dd7070Spatrick 
9*e5dd7070Spatrick #ifndef LLVM_CLANG_BASIC_PLISTSUPPORT_H
10*e5dd7070Spatrick #define LLVM_CLANG_BASIC_PLISTSUPPORT_H
11*e5dd7070Spatrick 
12*e5dd7070Spatrick #include "clang/Basic/LLVM.h"
13*e5dd7070Spatrick #include "clang/Basic/SourceLocation.h"
14*e5dd7070Spatrick #include "clang/Basic/SourceManager.h"
15*e5dd7070Spatrick #include "llvm/ADT/DenseMap.h"
16*e5dd7070Spatrick #include "llvm/ADT/SmallVector.h"
17*e5dd7070Spatrick #include "llvm/ADT/StringRef.h"
18*e5dd7070Spatrick #include "llvm/Support/raw_ostream.h"
19*e5dd7070Spatrick #include <cassert>
20*e5dd7070Spatrick #include <cstdint>
21*e5dd7070Spatrick 
22*e5dd7070Spatrick namespace clang {
23*e5dd7070Spatrick namespace markup {
24*e5dd7070Spatrick 
25*e5dd7070Spatrick using FIDMap = llvm::DenseMap<FileID, unsigned>;
26*e5dd7070Spatrick 
AddFID(FIDMap & FIDs,SmallVectorImpl<FileID> & V,FileID FID)27*e5dd7070Spatrick inline unsigned AddFID(FIDMap &FIDs, SmallVectorImpl<FileID> &V,
28*e5dd7070Spatrick                    FileID FID) {
29*e5dd7070Spatrick   FIDMap::iterator I = FIDs.find(FID);
30*e5dd7070Spatrick   if (I != FIDs.end())
31*e5dd7070Spatrick     return I->second;
32*e5dd7070Spatrick   unsigned NewValue = V.size();
33*e5dd7070Spatrick   FIDs[FID] = NewValue;
34*e5dd7070Spatrick   V.push_back(FID);
35*e5dd7070Spatrick   return NewValue;
36*e5dd7070Spatrick }
37*e5dd7070Spatrick 
AddFID(FIDMap & FIDs,SmallVectorImpl<FileID> & V,const SourceManager & SM,SourceLocation L)38*e5dd7070Spatrick inline unsigned AddFID(FIDMap &FIDs, SmallVectorImpl<FileID> &V,
39*e5dd7070Spatrick                    const SourceManager &SM, SourceLocation L) {
40*e5dd7070Spatrick   FileID FID = SM.getFileID(SM.getExpansionLoc(L));
41*e5dd7070Spatrick   return AddFID(FIDs, V, FID);
42*e5dd7070Spatrick }
43*e5dd7070Spatrick 
GetFID(const FIDMap & FIDs,FileID FID)44*e5dd7070Spatrick inline unsigned GetFID(const FIDMap &FIDs, FileID FID) {
45*e5dd7070Spatrick   FIDMap::const_iterator I = FIDs.find(FID);
46*e5dd7070Spatrick   assert(I != FIDs.end());
47*e5dd7070Spatrick   return I->second;
48*e5dd7070Spatrick }
49*e5dd7070Spatrick 
GetFID(const FIDMap & FIDs,const SourceManager & SM,SourceLocation L)50*e5dd7070Spatrick inline unsigned GetFID(const FIDMap &FIDs, const SourceManager &SM,
51*e5dd7070Spatrick                        SourceLocation L) {
52*e5dd7070Spatrick   FileID FID = SM.getFileID(SM.getExpansionLoc(L));
53*e5dd7070Spatrick   return GetFID(FIDs, FID);
54*e5dd7070Spatrick }
55*e5dd7070Spatrick 
Indent(raw_ostream & o,const unsigned indent)56*e5dd7070Spatrick inline raw_ostream &Indent(raw_ostream &o, const unsigned indent) {
57*e5dd7070Spatrick   for (unsigned i = 0; i < indent; ++i)
58*e5dd7070Spatrick     o << ' ';
59*e5dd7070Spatrick   return o;
60*e5dd7070Spatrick }
61*e5dd7070Spatrick 
EmitPlistHeader(raw_ostream & o)62*e5dd7070Spatrick inline raw_ostream &EmitPlistHeader(raw_ostream &o) {
63*e5dd7070Spatrick   static const char *PlistHeader =
64*e5dd7070Spatrick       "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
65*e5dd7070Spatrick       "<!DOCTYPE plist PUBLIC \"-//Apple Computer//DTD PLIST 1.0//EN\" "
66*e5dd7070Spatrick       "\"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n"
67*e5dd7070Spatrick       "<plist version=\"1.0\">\n";
68*e5dd7070Spatrick   return o << PlistHeader;
69*e5dd7070Spatrick }
70*e5dd7070Spatrick 
EmitInteger(raw_ostream & o,int64_t value)71*e5dd7070Spatrick inline raw_ostream &EmitInteger(raw_ostream &o, int64_t value) {
72*e5dd7070Spatrick   o << "<integer>";
73*e5dd7070Spatrick   o << value;
74*e5dd7070Spatrick   o << "</integer>";
75*e5dd7070Spatrick   return o;
76*e5dd7070Spatrick }
77*e5dd7070Spatrick 
EmitString(raw_ostream & o,StringRef s)78*e5dd7070Spatrick inline raw_ostream &EmitString(raw_ostream &o, StringRef s) {
79*e5dd7070Spatrick   o << "<string>";
80*e5dd7070Spatrick   for (StringRef::const_iterator I = s.begin(), E = s.end(); I != E; ++I) {
81*e5dd7070Spatrick     char c = *I;
82*e5dd7070Spatrick     switch (c) {
83*e5dd7070Spatrick     default:
84*e5dd7070Spatrick       o << c;
85*e5dd7070Spatrick       break;
86*e5dd7070Spatrick     case '&':
87*e5dd7070Spatrick       o << "&amp;";
88*e5dd7070Spatrick       break;
89*e5dd7070Spatrick     case '<':
90*e5dd7070Spatrick       o << "&lt;";
91*e5dd7070Spatrick       break;
92*e5dd7070Spatrick     case '>':
93*e5dd7070Spatrick       o << "&gt;";
94*e5dd7070Spatrick       break;
95*e5dd7070Spatrick     case '\'':
96*e5dd7070Spatrick       o << "&apos;";
97*e5dd7070Spatrick       break;
98*e5dd7070Spatrick     case '\"':
99*e5dd7070Spatrick       o << "&quot;";
100*e5dd7070Spatrick       break;
101*e5dd7070Spatrick     }
102*e5dd7070Spatrick   }
103*e5dd7070Spatrick   o << "</string>";
104*e5dd7070Spatrick   return o;
105*e5dd7070Spatrick }
106*e5dd7070Spatrick 
EmitLocation(raw_ostream & o,const SourceManager & SM,SourceLocation L,const FIDMap & FM,unsigned indent)107*e5dd7070Spatrick inline void EmitLocation(raw_ostream &o, const SourceManager &SM,
108*e5dd7070Spatrick                          SourceLocation L, const FIDMap &FM, unsigned indent) {
109*e5dd7070Spatrick   if (L.isInvalid()) return;
110*e5dd7070Spatrick 
111*e5dd7070Spatrick   FullSourceLoc Loc(SM.getExpansionLoc(L), const_cast<SourceManager &>(SM));
112*e5dd7070Spatrick 
113*e5dd7070Spatrick   Indent(o, indent) << "<dict>\n";
114*e5dd7070Spatrick   Indent(o, indent) << " <key>line</key>";
115*e5dd7070Spatrick   EmitInteger(o, Loc.getExpansionLineNumber()) << '\n';
116*e5dd7070Spatrick   Indent(o, indent) << " <key>col</key>";
117*e5dd7070Spatrick   EmitInteger(o, Loc.getExpansionColumnNumber()) << '\n';
118*e5dd7070Spatrick   Indent(o, indent) << " <key>file</key>";
119*e5dd7070Spatrick   EmitInteger(o, GetFID(FM, SM, Loc)) << '\n';
120*e5dd7070Spatrick   Indent(o, indent) << "</dict>\n";
121*e5dd7070Spatrick }
122*e5dd7070Spatrick 
EmitRange(raw_ostream & o,const SourceManager & SM,CharSourceRange R,const FIDMap & FM,unsigned indent)123*e5dd7070Spatrick inline void EmitRange(raw_ostream &o, const SourceManager &SM,
124*e5dd7070Spatrick                       CharSourceRange R, const FIDMap &FM, unsigned indent) {
125*e5dd7070Spatrick   if (R.isInvalid()) return;
126*e5dd7070Spatrick 
127*e5dd7070Spatrick   assert(R.isCharRange() && "cannot handle a token range");
128*e5dd7070Spatrick   Indent(o, indent) << "<array>\n";
129*e5dd7070Spatrick   EmitLocation(o, SM, R.getBegin(), FM, indent + 1);
130*e5dd7070Spatrick 
131*e5dd7070Spatrick   // The ".getLocWithOffset(-1)" emulates the behavior of an off-by-one bug
132*e5dd7070Spatrick   // in Lexer that is already fixed. It is here for backwards compatibility
133*e5dd7070Spatrick   // even though it is incorrect.
134*e5dd7070Spatrick   EmitLocation(o, SM, R.getEnd().getLocWithOffset(-1), FM, indent + 1);
135*e5dd7070Spatrick   Indent(o, indent) << "</array>\n";
136*e5dd7070Spatrick }
137*e5dd7070Spatrick 
138*e5dd7070Spatrick } // namespace markup
139*e5dd7070Spatrick } // namespace clang
140*e5dd7070Spatrick 
141*e5dd7070Spatrick #endif // LLVM_CLANG_BASIC_PLISTSUPPORT_H
142