| //===- CIndex.cpp - Clang-C Source Indexing Library -----------------------===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file implements the Clang-C Source Indexing library. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "CIndexer.h" |
| |
| #include "clang/AST/Decl.h" |
| #include "clang/AST/DeclVisitor.h" |
| #include "clang/AST/StmtVisitor.h" |
| #include "clang/Basic/FileManager.h" |
| #include "clang/Basic/SourceManager.h" |
| #include "clang/Basic/Version.h" |
| #include "clang/Sema/CodeCompleteConsumer.h" |
| #include "llvm/ADT/StringExtras.h" |
| #include "llvm/Config/config.h" |
| #include "llvm/Support/Compiler.h" |
| #include "llvm/Support/MemoryBuffer.h" |
| #include "llvm/Support/raw_ostream.h" |
| #include "llvm/Support/Program.h" |
| |
| #include <cstdio> |
| #include <vector> |
| #include <sstream> |
| |
| #ifdef __CYGWIN__ |
| #include <cygwin/version.h> |
| #include <sys/cygwin.h> |
| #define LLVM_ON_WIN32 1 |
| #endif |
| |
| #ifdef LLVM_ON_WIN32 |
| #include <windows.h> |
| #else |
| #include <dlfcn.h> |
| #endif |
| |
| using namespace clang; |
| |
| std::string CIndexer::getClangResourcesPath() { |
| // Did we already compute the path? |
| if (!ResourcesPath.empty()) |
| return ResourcesPath.str(); |
| |
| // Find the location where this library lives (libclang.dylib). |
| #ifdef LLVM_ON_WIN32 |
| MEMORY_BASIC_INFORMATION mbi; |
| char path[MAX_PATH]; |
| VirtualQuery((void *)(uintptr_t)clang_createTranslationUnit, &mbi, |
| sizeof(mbi)); |
| GetModuleFileNameA((HINSTANCE)mbi.AllocationBase, path, MAX_PATH); |
| |
| #ifdef __CYGWIN__ |
| char w32path[MAX_PATH]; |
| strcpy(w32path, path); |
| #if CYGWIN_VERSION_API_MAJOR > 0 || CYGWIN_VERSION_API_MINOR >= 181 |
| cygwin_conv_path(CCP_WIN_A_TO_POSIX, w32path, path, MAX_PATH); |
| #else |
| cygwin_conv_to_full_posix_path(w32path, path); |
| #endif |
| #endif |
| |
| llvm::sys::Path LibClangPath(path); |
| LibClangPath.eraseComponent(); |
| #else |
| // This silly cast below avoids a C++ warning. |
| Dl_info info; |
| if (dladdr((void *)(uintptr_t)clang_createTranslationUnit, &info) == 0) |
| llvm_unreachable("Call to dladdr() failed"); |
| |
| llvm::sys::Path LibClangPath(info.dli_fname); |
| |
| // We now have the CIndex directory, locate clang relative to it. |
| LibClangPath.eraseComponent(); |
| #endif |
| |
| LibClangPath.appendComponent("clang"); |
| LibClangPath.appendComponent(CLANG_VERSION_STRING); |
| |
| // Cache our result. |
| ResourcesPath = LibClangPath; |
| return LibClangPath.str(); |
| } |
| |
| static llvm::sys::Path GetTemporaryPath() { |
| // FIXME: This is lame; sys::Path should provide this function (in particular, |
| // it should know how to find the temporary files dir). |
| std::string Error; |
| const char *TmpDir = ::getenv("TMPDIR"); |
| if (!TmpDir) |
| TmpDir = ::getenv("TEMP"); |
| if (!TmpDir) |
| TmpDir = ::getenv("TMP"); |
| if (!TmpDir) |
| TmpDir = "/tmp"; |
| llvm::sys::Path P(TmpDir); |
| P.appendComponent("remap"); |
| if (P.makeUnique(false, &Error)) |
| return llvm::sys::Path(""); |
| |
| // FIXME: Grumble, makeUnique sometimes leaves the file around!? PR3837. |
| P.eraseFromDisk(false, 0); |
| |
| return P; |
| } |
| |
| bool clang::RemapFiles(unsigned num_unsaved_files, |
| struct CXUnsavedFile *unsaved_files, |
| std::vector<std::string> &RemapArgs, |
| std::vector<llvm::sys::Path> &TemporaryFiles) { |
| for (unsigned i = 0; i != num_unsaved_files; ++i) { |
| // Write the contents of this unsaved file into the temporary file. |
| llvm::sys::Path SavedFile(GetTemporaryPath()); |
| if (SavedFile.empty()) |
| return true; |
| |
| std::string ErrorInfo; |
| llvm::raw_fd_ostream OS(SavedFile.c_str(), ErrorInfo, |
| llvm::raw_fd_ostream::F_Binary); |
| if (!ErrorInfo.empty()) |
| return true; |
| |
| OS.write(unsaved_files[i].Contents, unsaved_files[i].Length); |
| OS.close(); |
| if (OS.has_error()) { |
| SavedFile.eraseFromDisk(); |
| OS.clear_error(); |
| return true; |
| } |
| |
| // Remap the file. |
| std::string RemapArg = unsaved_files[i].Filename; |
| RemapArg += ';'; |
| RemapArg += SavedFile.str(); |
| RemapArgs.push_back("-Xclang"); |
| RemapArgs.push_back("-remap-file"); |
| RemapArgs.push_back("-Xclang"); |
| RemapArgs.push_back(RemapArg); |
| TemporaryFiles.push_back(SavedFile); |
| } |
| |
| return false; |
| } |
| |