RewriterTestContext.h revision 99eb4a7920667d23a89df8fcc6429d12b36e4d57
1//===--- RewriterTestContext.h ----------------------------------*- 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 a utility class for Rewriter related tests.
11//
12//===----------------------------------------------------------------------===//
13
14#ifndef LLVM_CLANG_REWRITER_TEST_CONTEXT_H
15#define LLVM_CLANG_REWRITER_TEST_CONTEXT_H
16
17#include "clang/Basic/Diagnostic.h"
18#include "clang/Basic/DiagnosticOptions.h"
19#include "clang/Basic/FileManager.h"
20#include "clang/Basic/LangOptions.h"
21#include "clang/Basic/SourceManager.h"
22#include "clang/Frontend/TextDiagnosticPrinter.h"
23#include "clang/Rewrite/Core/Rewriter.h"
24#include "llvm/Support/FileSystem.h"
25#include "llvm/Support/Path.h"
26#include "llvm/Support/raw_ostream.h"
27
28namespace clang {
29
30/// \brief A class that sets up a ready to use Rewriter.
31///
32/// Useful in unit tests that need a Rewriter. Creates all dependencies
33/// of a Rewriter with default values for testing and provides convenience
34/// methods, which help with writing tests that change files.
35class RewriterTestContext {
36 public:
37  RewriterTestContext()
38      : DiagOpts(new DiagnosticOptions()),
39        Diagnostics(llvm::IntrusiveRefCntPtr<DiagnosticIDs>(new DiagnosticIDs),
40                    &*DiagOpts),
41        DiagnosticPrinter(llvm::outs(), &*DiagOpts),
42        Files((FileSystemOptions())),
43        Sources(Diagnostics, Files),
44        Rewrite(Sources, Options) {
45    Diagnostics.setClient(&DiagnosticPrinter, false);
46  }
47
48  ~RewriterTestContext() {
49    if (!TemporaryDirectory.empty()) {
50      uint32_t RemovedCount = 0;
51      llvm::sys::fs::remove_all(TemporaryDirectory.str(), RemovedCount);
52    }
53  }
54
55  FileID createInMemoryFile(StringRef Name, StringRef Content) {
56    const llvm::MemoryBuffer *Source =
57      llvm::MemoryBuffer::getMemBuffer(Content);
58    const FileEntry *Entry =
59      Files.getVirtualFile(Name, Source->getBufferSize(), 0);
60    Sources.overrideFileContents(Entry, Source, true);
61    assert(Entry != NULL);
62    return Sources.createFileID(Entry, SourceLocation(), SrcMgr::C_User);
63  }
64
65  FileID createOnDiskFile(StringRef Name, StringRef Content) {
66    if (TemporaryDirectory.empty()) {
67      int FD;
68      bool error =
69        llvm::sys::fs::unique_file("rewriter-test-%%-%%-%%-%%/anchor", FD,
70                                   TemporaryDirectory);
71      assert(!error); (void)error;
72      llvm::raw_fd_ostream Closer(FD, /*shouldClose=*/true);
73      TemporaryDirectory = llvm::sys::path::parent_path(TemporaryDirectory);
74    }
75    llvm::SmallString<1024> Path(TemporaryDirectory);
76    llvm::sys::path::append(Path, Name);
77    std::string ErrorInfo;
78    llvm::raw_fd_ostream OutStream(Path.c_str(),
79                                   ErrorInfo, llvm::raw_fd_ostream::F_Binary);
80    assert(ErrorInfo.empty());
81    OutStream << Content;
82    OutStream.close();
83    const FileEntry *File = Files.getFile(Path);
84    assert(File != NULL);
85    return Sources.createFileID(File, SourceLocation(), SrcMgr::C_User);
86  }
87
88  SourceLocation getLocation(FileID ID, unsigned Line, unsigned Column) {
89    SourceLocation Result = Sources.translateFileLineCol(
90        Sources.getFileEntryForID(ID), Line, Column);
91    assert(Result.isValid());
92    return Result;
93  }
94
95  std::string getRewrittenText(FileID ID) {
96    std::string Result;
97    llvm::raw_string_ostream OS(Result);
98    Rewrite.getEditBuffer(ID).write(OS);
99    OS.flush();
100    return Result;
101  }
102
103  std::string getFileContentFromDisk(StringRef Name) {
104    llvm::SmallString<1024> Path(TemporaryDirectory.str());
105    llvm::sys::path::append(Path, Name);
106    // We need to read directly from the FileManager without relaying through
107    // a FileEntry, as otherwise we'd read through an already opened file
108    // descriptor, which might not see the changes made.
109    // FIXME: Figure out whether there is a way to get the SourceManger to
110    // reopen the file.
111    return Files.getBufferForFile(Path, NULL)->getBuffer();
112  }
113
114  llvm::IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts;
115  DiagnosticsEngine Diagnostics;
116  TextDiagnosticPrinter DiagnosticPrinter;
117  FileManager Files;
118  SourceManager Sources;
119  LangOptions Options;
120  Rewriter Rewrite;
121
122  // Will be set once on disk files are generated.
123  SmallString<128> TemporaryDirectory;
124};
125
126} // end namespace clang
127
128#endif
129