diff --git a/unittests/libclang/LibclangTest.cpp b/unittests/libclang/LibclangTest.cpp index 1fb7f434349..899afd0380e 100644 --- a/unittests/libclang/LibclangTest.cpp +++ b/unittests/libclang/LibclangTest.cpp @@ -14,6 +14,7 @@ #include "llvm/Support/Path.h" #include "llvm/Support/raw_ostream.h" #include "gtest/gtest.h" +#include "TestUtils.h" #include #include #include @@ -353,77 +354,6 @@ TEST(libclang, ModuleMapDescriptor) { clang_ModuleMapDescriptor_dispose(MMD); } -class LibclangParseTest : public ::testing::Test { - std::set Files; - typedef std::unique_ptr fixed_addr_string; - std::map UnsavedFileContents; -public: - std::string TestDir; - CXIndex Index; - CXTranslationUnit ClangTU; - unsigned TUFlags; - std::vector UnsavedFiles; - - void SetUp() override { - llvm::SmallString<256> Dir; - ASSERT_FALSE(llvm::sys::fs::createUniqueDirectory("libclang-test", Dir)); - TestDir = Dir.str(); - TUFlags = CXTranslationUnit_DetailedPreprocessingRecord | - clang_defaultEditingTranslationUnitOptions(); - Index = clang_createIndex(0, 0); - ClangTU = nullptr; - } - void TearDown() override { - clang_disposeTranslationUnit(ClangTU); - clang_disposeIndex(Index); - for (const std::string &Path : Files) - llvm::sys::fs::remove(Path); - llvm::sys::fs::remove(TestDir); - } - void WriteFile(std::string &Filename, const std::string &Contents) { - if (!llvm::sys::path::is_absolute(Filename)) { - llvm::SmallString<256> Path(TestDir); - llvm::sys::path::append(Path, Filename); - Filename = Path.str(); - Files.insert(Filename); - } - llvm::sys::fs::create_directories(llvm::sys::path::parent_path(Filename)); - std::ofstream OS(Filename); - OS << Contents; - assert(OS.good()); - } - void MapUnsavedFile(std::string Filename, const std::string &Contents) { - if (!llvm::sys::path::is_absolute(Filename)) { - llvm::SmallString<256> Path(TestDir); - llvm::sys::path::append(Path, Filename); - Filename = Path.str(); - } - auto it = UnsavedFileContents.insert(std::make_pair( - fixed_addr_string(new std::string(Filename)), - fixed_addr_string(new std::string(Contents)))); - UnsavedFiles.push_back({ - it.first->first->c_str(), // filename - it.first->second->c_str(), // contents - it.first->second->size() // length - }); - } - template - void Traverse(const F &TraversalFunctor) { - CXCursor TuCursor = clang_getTranslationUnitCursor(ClangTU); - std::reference_wrapper FunctorRef = std::cref(TraversalFunctor); - clang_visitChildren(TuCursor, - &TraverseStateless>, - &FunctorRef); - } -private: - template - static CXChildVisitResult TraverseStateless(CXCursor cx, CXCursor parent, - CXClientData data) { - TState *State = static_cast(data); - return State->get()(cx, parent); - } -}; - TEST_F(LibclangParseTest, AllSkippedRanges) { std::string Header = "header.h", Main = "main.cpp"; WriteFile(Header, diff --git a/unittests/libclang/TestUtils.h b/unittests/libclang/TestUtils.h new file mode 100644 index 00000000000..883155ec81f --- /dev/null +++ b/unittests/libclang/TestUtils.h @@ -0,0 +1,94 @@ +//===- unittests/libclang/TestUtils.h -------------------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CLANG_TEST_TESTUTILS_H +#define LLVM_CLANG_TEST_TESTUTILS_H + +#include "clang-c/Index.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/Support/FileSystem.h" +#include "llvm/Support/Path.h" + +#include +#include +#include +#include +#include "gtest/gtest.h" + +class LibclangParseTest : public ::testing::Test { + std::set Files; + typedef std::unique_ptr fixed_addr_string; + std::map UnsavedFileContents; +public: + std::string TestDir; + CXIndex Index; + CXTranslationUnit ClangTU; + unsigned TUFlags; + std::vector UnsavedFiles; + + void SetUp() override { + llvm::SmallString<256> Dir; + ASSERT_FALSE(llvm::sys::fs::createUniqueDirectory("libclang-test", Dir)); + TestDir = Dir.str(); + TUFlags = CXTranslationUnit_DetailedPreprocessingRecord | + clang_defaultEditingTranslationUnitOptions(); + Index = clang_createIndex(0, 0); + ClangTU = nullptr; + } + void TearDown() override { + clang_disposeTranslationUnit(ClangTU); + clang_disposeIndex(Index); + for (const std::string &Path : Files) + llvm::sys::fs::remove(Path); + llvm::sys::fs::remove(TestDir); + } + void WriteFile(std::string &Filename, const std::string &Contents) { + if (!llvm::sys::path::is_absolute(Filename)) { + llvm::SmallString<256> Path(TestDir); + llvm::sys::path::append(Path, Filename); + Filename = Path.str(); + Files.insert(Filename); + } + llvm::sys::fs::create_directories(llvm::sys::path::parent_path(Filename)); + std::ofstream OS(Filename); + OS << Contents; + assert(OS.good()); + } + void MapUnsavedFile(std::string Filename, const std::string &Contents) { + if (!llvm::sys::path::is_absolute(Filename)) { + llvm::SmallString<256> Path(TestDir); + llvm::sys::path::append(Path, Filename); + Filename = Path.str(); + } + auto it = UnsavedFileContents.insert(std::make_pair( + fixed_addr_string(new std::string(Filename)), + fixed_addr_string(new std::string(Contents)))); + UnsavedFiles.push_back({ + it.first->first->c_str(), // filename + it.first->second->c_str(), // contents + it.first->second->size() // length + }); + } + template + void Traverse(const F &TraversalFunctor) { + CXCursor TuCursor = clang_getTranslationUnitCursor(ClangTU); + std::reference_wrapper FunctorRef = std::cref(TraversalFunctor); + clang_visitChildren(TuCursor, + &TraverseStateless>, + &FunctorRef); + } +private: + template + static CXChildVisitResult TraverseStateless(CXCursor cx, CXCursor parent, + CXClientData data) { + TState *State = static_cast(data); + return State->get()(cx, parent); + } +}; + +#endif // LLVM_CLANG_TEST_TESTUTILS_H \ No newline at end of file