Files
swift-mirror/lib/Basic/Version.cpp
Jordan Rose b7bc622f63 If there's no SWIFT_COMPILER_VERSION, show the revisions Swift was built with.
Modeled after the same feature in Clang. Example (from swift -version):

  Swift version 2.1 (LLVM bf75799360, Clang 340dfbb3d5, Swift 32557)

rdar://problem/22959963

Swift SVN r32625
2015-10-12 16:32:52 +00:00

190 lines
5.5 KiB
C++

//===--- Version.cpp - Swift Version Number -------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See http://swift.org/LICENSE.txt for license information
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This file defines several version-related utility functions for Swift.
//
//===----------------------------------------------------------------------===//
#include "clang/Basic/CharInfo.h"
#include "llvm/Support/raw_ostream.h"
#include "llvm/ADT/SmallString.h"
#include "swift/AST/DiagnosticsParse.h"
#include "swift/Basic/LLVM.h"
#include "swift/Basic/Version.h"
#include <vector>
#define TOSTR2(X) #X
#define TOSTR(X) TOSTR2(X)
#ifdef SWIFT_VERSION_PATCHLEVEL
/// Helper macro for SWIFT_VERSION_STRING.
#define SWIFT_MAKE_VERSION_STRING(X, Y, Z) TOSTR(X) "." TOSTR(Y) "." TOSTR(Z)
/// A string that describes the Swift version number, e.g., "1.0".
#define SWIFT_VERSION_STRING \
SWIFT_MAKE_VERSION_STRING(SWIFT_VERSION_MAJOR, SWIFT_VERSION_MINOR, \
SWIFT_VERSION_PATCHLEVEL)
#else
/// Helper macro for SWIFT_VERSION_STRING.
#define SWIFT_MAKE_VERSION_STRING(X, Y) TOSTR(X) "." TOSTR(Y)
/// A string that describes the Swift version number, e.g., "1.0".
#define SWIFT_VERSION_STRING \
SWIFT_MAKE_VERSION_STRING(SWIFT_VERSION_MAJOR, SWIFT_VERSION_MINOR)
#endif
#if __has_include("LLVMRevision.inc")
# include "LLVMRevision.inc"
#endif
#if __has_include("ClangRevision.inc")
# include "ClangRevision.inc"
#endif
#if __has_include("SwiftRevision.inc")
# include "SwiftRevision.inc"
#endif
namespace swift {
namespace version {
/// Print a string of the form "LLVM xxxxx, Clang yyyyy, Swift zzzzz",
/// where each placeholder is the revision for the associated repository.
static void printFullRevisionString(raw_ostream &out) {
// Abitrarily truncate to 10 characters. This should be enough to unique
// Git hashes for the time being, and certainly enough for SVN revisions,
// while keeping the version string from being ridiculously long.
#if defined(LLVM_REVISION)
out << "LLVM " << StringRef(LLVM_REVISION).slice(0, 10);
# if defined(CLANG_REVISION) || defined(SWIFT_REVISION)
out << ", ";
# endif
#endif
#if defined(CLANG_REVISION)
out << "Clang " << StringRef(CLANG_REVISION).slice(0, 10);
# if defined(SWIFT_REVISION)
out << ", ";
# endif
#endif
#if defined(SWIFT_REVISION)
out << "Swift " << StringRef(SWIFT_REVISION).slice(0, 10);
#endif
}
void parseVersionString(StringRef VersionString,
SmallVectorImpl<unsigned> &Components,
SourceLoc Loc,
DiagnosticEngine *Diags) {
SmallString<16> digits;
llvm::raw_svector_ostream OS(digits);
unsigned Component;
for (auto c : VersionString) {
if (Loc.isValid())
Loc = Loc.getAdvancedLoc(1);
if (clang::isDigit(c)) {
OS << c;
} else if (c == '.' && OS.str().size()) {
OS.str().getAsInteger(10, Component);
Components.push_back(Component);
digits.clear();
} else {
Components.clear();
if (Diags)
Diags->diagnose(Loc, diag::invalid_character_in_compiler_version);
else
llvm_unreachable("Invalid character in _compiler_version build configuration");
}
}
if (OS.str().size()) {
OS.str().getAsInteger(10, Component);
Components.push_back(Component);
}
}
CompilerVersion::CompilerVersion(const StringRef VersionString,
SourceLoc Loc, DiagnosticEngine *Diags) {
parseVersionString(VersionString, Components, Loc, Diags);
}
CompilerVersion::CompilerVersion() {
#ifdef SWIFT_COMPILER_VERSION
parseVersionString(TOSTR(SWIFT_COMPILER_VERSION), Components, SourceLoc(),
nullptr);
#endif
}
std::string CompilerVersion::str() const {
std::string VersionString;
llvm::raw_string_ostream OS(VersionString);
for (auto i = Components.begin(); i != Components.end(); i++) {
OS << *i;
if (i != Components.end() - 1)
OS << '.';
}
return OS.str();
}
bool operator>=(const class CompilerVersion &lhs,
const class CompilerVersion &rhs) {
// The empty compiler version represents the latest possible version,
// usually built from the source repository.
if (lhs.empty())
return true;
auto n = std::min(lhs.size(), rhs.size());
for (size_t i = 0; i < n; ++i) {
// Skip the second component for comparison - this is just an internal
// compiler variant.
if (i == 1)
continue;
if (lhs[i] < rhs[i])
return false;
else if (lhs[i] > rhs[i])
return true;
}
return lhs.size() >= rhs.size();
}
std::pair<unsigned, unsigned> getSwiftNumericVersion() {
return { SWIFT_VERSION_MAJOR, SWIFT_VERSION_MINOR };
}
std::string getSwiftFullVersion() {
std::string buf;
llvm::raw_string_ostream OS(buf);
#ifdef SWIFT_VENDOR
OS << SWIFT_VENDOR " ";
#endif
OS << "Swift version " SWIFT_VERSION_STRING;
#if defined(SWIFT_COMPILER_VERSION)
OS << " (" TOSTR(SWIFT_COMPILER_VERSION) ")";
#elif defined(LLVM_REVISION) || defined(CLANG_REVISION) || \
defined(SWIFT_REVISION)
OS << " (";
printFullRevisionString(OS);
OS << ")";
#endif
return OS.str();
}
} // end namespace version
} // end namespace swift