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 << "&";
88*e5dd7070Spatrick break;
89*e5dd7070Spatrick case '<':
90*e5dd7070Spatrick o << "<";
91*e5dd7070Spatrick break;
92*e5dd7070Spatrick case '>':
93*e5dd7070Spatrick o << ">";
94*e5dd7070Spatrick break;
95*e5dd7070Spatrick case '\'':
96*e5dd7070Spatrick o << "'";
97*e5dd7070Spatrick break;
98*e5dd7070Spatrick case '\"':
99*e5dd7070Spatrick o << """;
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