1*0b57cec5SDimitry Andric //===- Version.cpp - Clang Version Number -----------------------*- C++ -*-===// 2*0b57cec5SDimitry Andric // 3*0b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*0b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5*0b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*0b57cec5SDimitry Andric // 7*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 8*0b57cec5SDimitry Andric // 9*0b57cec5SDimitry Andric // This file defines several version-related utility functions for Clang. 10*0b57cec5SDimitry Andric // 11*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 12*0b57cec5SDimitry Andric 13*0b57cec5SDimitry Andric #include "clang/Basic/Version.h" 14*0b57cec5SDimitry Andric #include "clang/Basic/LLVM.h" 15*0b57cec5SDimitry Andric #include "clang/Config/config.h" 16*0b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h" 17*0b57cec5SDimitry Andric #include <cstdlib> 18*0b57cec5SDimitry Andric #include <cstring> 19*0b57cec5SDimitry Andric 20*0b57cec5SDimitry Andric #ifdef HAVE_VCS_VERSION_INC 21*0b57cec5SDimitry Andric #include "VCSVersion.inc" 22*0b57cec5SDimitry Andric #endif 23*0b57cec5SDimitry Andric 24*0b57cec5SDimitry Andric namespace clang { 25*0b57cec5SDimitry Andric 26*0b57cec5SDimitry Andric std::string getClangRepositoryPath() { 27*0b57cec5SDimitry Andric #if defined(CLANG_REPOSITORY_STRING) 28*0b57cec5SDimitry Andric return CLANG_REPOSITORY_STRING; 29*0b57cec5SDimitry Andric #else 30*0b57cec5SDimitry Andric #ifdef CLANG_REPOSITORY 31*0b57cec5SDimitry Andric StringRef URL(CLANG_REPOSITORY); 32*0b57cec5SDimitry Andric #else 33*0b57cec5SDimitry Andric StringRef URL(""); 34*0b57cec5SDimitry Andric #endif 35*0b57cec5SDimitry Andric 36*0b57cec5SDimitry Andric // If the CLANG_REPOSITORY is empty, try to use the SVN keyword. This helps us 37*0b57cec5SDimitry Andric // pick up a tag in an SVN export, for example. 38*0b57cec5SDimitry Andric StringRef SVNRepository("$URL: https://llvm.org/svn/llvm-project/cfe/tags/RELEASE_900/final/lib/Basic/Version.cpp $"); 39*0b57cec5SDimitry Andric if (URL.empty()) { 40*0b57cec5SDimitry Andric URL = SVNRepository.slice(SVNRepository.find(':'), 41*0b57cec5SDimitry Andric SVNRepository.find("/lib/Basic")); 42*0b57cec5SDimitry Andric } 43*0b57cec5SDimitry Andric 44*0b57cec5SDimitry Andric // Strip off version from a build from an integration branch. 45*0b57cec5SDimitry Andric URL = URL.slice(0, URL.find("/src/tools/clang")); 46*0b57cec5SDimitry Andric 47*0b57cec5SDimitry Andric // Trim path prefix off, assuming path came from standard cfe path. 48*0b57cec5SDimitry Andric size_t Start = URL.find("cfe/"); 49*0b57cec5SDimitry Andric if (Start != StringRef::npos) 50*0b57cec5SDimitry Andric URL = URL.substr(Start + 4); 51*0b57cec5SDimitry Andric 52*0b57cec5SDimitry Andric return URL; 53*0b57cec5SDimitry Andric #endif 54*0b57cec5SDimitry Andric } 55*0b57cec5SDimitry Andric 56*0b57cec5SDimitry Andric std::string getLLVMRepositoryPath() { 57*0b57cec5SDimitry Andric #ifdef LLVM_REPOSITORY 58*0b57cec5SDimitry Andric StringRef URL(LLVM_REPOSITORY); 59*0b57cec5SDimitry Andric #else 60*0b57cec5SDimitry Andric StringRef URL(""); 61*0b57cec5SDimitry Andric #endif 62*0b57cec5SDimitry Andric 63*0b57cec5SDimitry Andric // Trim path prefix off, assuming path came from standard llvm path. 64*0b57cec5SDimitry Andric // Leave "llvm/" prefix to distinguish the following llvm revision from the 65*0b57cec5SDimitry Andric // clang revision. 66*0b57cec5SDimitry Andric size_t Start = URL.find("llvm/"); 67*0b57cec5SDimitry Andric if (Start != StringRef::npos) 68*0b57cec5SDimitry Andric URL = URL.substr(Start); 69*0b57cec5SDimitry Andric 70*0b57cec5SDimitry Andric return URL; 71*0b57cec5SDimitry Andric } 72*0b57cec5SDimitry Andric 73*0b57cec5SDimitry Andric std::string getClangRevision() { 74*0b57cec5SDimitry Andric #ifdef CLANG_REVISION 75*0b57cec5SDimitry Andric return CLANG_REVISION; 76*0b57cec5SDimitry Andric #else 77*0b57cec5SDimitry Andric return ""; 78*0b57cec5SDimitry Andric #endif 79*0b57cec5SDimitry Andric } 80*0b57cec5SDimitry Andric 81*0b57cec5SDimitry Andric std::string getLLVMRevision() { 82*0b57cec5SDimitry Andric #ifdef LLVM_REVISION 83*0b57cec5SDimitry Andric return LLVM_REVISION; 84*0b57cec5SDimitry Andric #else 85*0b57cec5SDimitry Andric return ""; 86*0b57cec5SDimitry Andric #endif 87*0b57cec5SDimitry Andric } 88*0b57cec5SDimitry Andric 89*0b57cec5SDimitry Andric std::string getClangFullRepositoryVersion() { 90*0b57cec5SDimitry Andric std::string buf; 91*0b57cec5SDimitry Andric llvm::raw_string_ostream OS(buf); 92*0b57cec5SDimitry Andric std::string Path = getClangRepositoryPath(); 93*0b57cec5SDimitry Andric std::string Revision = getClangRevision(); 94*0b57cec5SDimitry Andric if (!Path.empty() || !Revision.empty()) { 95*0b57cec5SDimitry Andric OS << '('; 96*0b57cec5SDimitry Andric if (!Path.empty()) 97*0b57cec5SDimitry Andric OS << Path; 98*0b57cec5SDimitry Andric if (!Revision.empty()) { 99*0b57cec5SDimitry Andric if (!Path.empty()) 100*0b57cec5SDimitry Andric OS << ' '; 101*0b57cec5SDimitry Andric OS << Revision; 102*0b57cec5SDimitry Andric } 103*0b57cec5SDimitry Andric OS << ')'; 104*0b57cec5SDimitry Andric } 105*0b57cec5SDimitry Andric // Support LLVM in a separate repository. 106*0b57cec5SDimitry Andric std::string LLVMRev = getLLVMRevision(); 107*0b57cec5SDimitry Andric if (!LLVMRev.empty() && LLVMRev != Revision) { 108*0b57cec5SDimitry Andric OS << " ("; 109*0b57cec5SDimitry Andric std::string LLVMRepo = getLLVMRepositoryPath(); 110*0b57cec5SDimitry Andric if (!LLVMRepo.empty()) 111*0b57cec5SDimitry Andric OS << LLVMRepo << ' '; 112*0b57cec5SDimitry Andric OS << LLVMRev << ')'; 113*0b57cec5SDimitry Andric } 114*0b57cec5SDimitry Andric return OS.str(); 115*0b57cec5SDimitry Andric } 116*0b57cec5SDimitry Andric 117*0b57cec5SDimitry Andric std::string getClangFullVersion() { 118*0b57cec5SDimitry Andric return getClangToolFullVersion("clang"); 119*0b57cec5SDimitry Andric } 120*0b57cec5SDimitry Andric 121*0b57cec5SDimitry Andric std::string getClangToolFullVersion(StringRef ToolName) { 122*0b57cec5SDimitry Andric std::string buf; 123*0b57cec5SDimitry Andric llvm::raw_string_ostream OS(buf); 124*0b57cec5SDimitry Andric #ifdef CLANG_VENDOR 125*0b57cec5SDimitry Andric OS << CLANG_VENDOR; 126*0b57cec5SDimitry Andric #endif 127*0b57cec5SDimitry Andric OS << ToolName << " version " CLANG_VERSION_STRING " " 128*0b57cec5SDimitry Andric << getClangFullRepositoryVersion(); 129*0b57cec5SDimitry Andric 130*0b57cec5SDimitry Andric // If vendor supplied, include the base LLVM version as well. 131*0b57cec5SDimitry Andric #ifdef CLANG_VENDOR 132*0b57cec5SDimitry Andric OS << " (based on " << BACKEND_PACKAGE_STRING << ")"; 133*0b57cec5SDimitry Andric #endif 134*0b57cec5SDimitry Andric 135*0b57cec5SDimitry Andric return OS.str(); 136*0b57cec5SDimitry Andric } 137*0b57cec5SDimitry Andric 138*0b57cec5SDimitry Andric std::string getClangFullCPPVersion() { 139*0b57cec5SDimitry Andric // The version string we report in __VERSION__ is just a compacted version of 140*0b57cec5SDimitry Andric // the one we report on the command line. 141*0b57cec5SDimitry Andric std::string buf; 142*0b57cec5SDimitry Andric llvm::raw_string_ostream OS(buf); 143*0b57cec5SDimitry Andric #ifdef CLANG_VENDOR 144*0b57cec5SDimitry Andric OS << CLANG_VENDOR; 145*0b57cec5SDimitry Andric #endif 146*0b57cec5SDimitry Andric OS << "Clang " CLANG_VERSION_STRING " " << getClangFullRepositoryVersion(); 147*0b57cec5SDimitry Andric return OS.str(); 148*0b57cec5SDimitry Andric } 149*0b57cec5SDimitry Andric 150*0b57cec5SDimitry Andric } // end namespace clang 151