#include "clang/Driver/Distro.h"
#include "clang/Basic/LLVM.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/ADT/StringRef.h"
#include "llvm/ADT/StringSwitch.h"
#include "llvm/ADT/Triple.h"
#include "llvm/Support/ErrorOr.h"
#include "llvm/Support/Host.h"
#include "llvm/Support/MemoryBuffer.h"
#include "llvm/Support/Threading.h"
using namespace clang::driver;
using namespace clang;
static Distro::DistroType DetectOsRelease(llvm::vfs::FileSystem &VFS) {
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> File =
VFS.getBufferForFile("/etc/os-release");
if (!File)
File = VFS.getBufferForFile("/usr/lib/os-release");
if (!File)
return Distro::UnknownDistro;
SmallVector<StringRef, 16> Lines;
File.get()->getBuffer().split(Lines, "\n");
Distro::DistroType Version = Distro::UnknownDistro;
for (StringRef Line : Lines)
if (Version == Distro::UnknownDistro && Line.startswith("ID="))
Version = llvm::StringSwitch<Distro::DistroType>(Line.substr(3))
.Case("alpine", Distro::AlpineLinux)
.Case("fedora", Distro::Fedora)
.Case("gentoo", Distro::Gentoo)
.Case("arch", Distro::ArchLinux)
.Case("sles", Distro::OpenSUSE)
.Case("opensuse", Distro::OpenSUSE)
.Case("exherbo", Distro::Exherbo)
.Default(Distro::UnknownDistro);
return Version;
}
static Distro::DistroType DetectLsbRelease(llvm::vfs::FileSystem &VFS) {
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> File =
VFS.getBufferForFile("/etc/lsb-release");
if (!File)
return Distro::UnknownDistro;
SmallVector<StringRef, 16> Lines;
File.get()->getBuffer().split(Lines, "\n");
Distro::DistroType Version = Distro::UnknownDistro;
for (StringRef Line : Lines)
if (Version == Distro::UnknownDistro &&
Line.startswith("DISTRIB_CODENAME="))
Version = llvm::StringSwitch<Distro::DistroType>(Line.substr(17))
.Case("hardy", Distro::UbuntuHardy)
.Case("intrepid", Distro::UbuntuIntrepid)
.Case("jaunty", Distro::UbuntuJaunty)
.Case("karmic", Distro::UbuntuKarmic)
.Case("lucid", Distro::UbuntuLucid)
.Case("maverick", Distro::UbuntuMaverick)
.Case("natty", Distro::UbuntuNatty)
.Case("oneiric", Distro::UbuntuOneiric)
.Case("precise", Distro::UbuntuPrecise)
.Case("quantal", Distro::UbuntuQuantal)
.Case("raring", Distro::UbuntuRaring)
.Case("saucy", Distro::UbuntuSaucy)
.Case("trusty", Distro::UbuntuTrusty)
.Case("utopic", Distro::UbuntuUtopic)
.Case("vivid", Distro::UbuntuVivid)
.Case("wily", Distro::UbuntuWily)
.Case("xenial", Distro::UbuntuXenial)
.Case("yakkety", Distro::UbuntuYakkety)
.Case("zesty", Distro::UbuntuZesty)
.Case("artful", Distro::UbuntuArtful)
.Case("bionic", Distro::UbuntuBionic)
.Case("cosmic", Distro::UbuntuCosmic)
.Case("disco", Distro::UbuntuDisco)
.Case("eoan", Distro::UbuntuEoan)
.Case("focal", Distro::UbuntuFocal)
.Case("groovy", Distro::UbuntuGroovy)
.Case("hirsute", Distro::UbuntuHirsute)
.Case("impish", Distro::UbuntuImpish)
.Case("jammy", Distro::UbuntuJammy)
.Case("kinetic", Distro::UbuntuKinetic)
.Default(Distro::UnknownDistro);
return Version;
}
static Distro::DistroType DetectDistro(llvm::vfs::FileSystem &VFS) {
Distro::DistroType Version = Distro::UnknownDistro;
Version = DetectOsRelease(VFS);
if (Version != Distro::UnknownDistro)
return Version;
Version = DetectLsbRelease(VFS);
if (Version != Distro::UnknownDistro)
return Version;
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> File =
VFS.getBufferForFile("/etc/redhat-release");
if (File) {
StringRef Data = File.get()->getBuffer();
if (Data.startswith("Fedora release"))
return Distro::Fedora;
if (Data.startswith("Red Hat Enterprise Linux") ||
Data.startswith("CentOS") || Data.startswith("Scientific Linux")) {
if (Data.contains("release 7"))
return Distro::RHEL7;
else if (Data.contains("release 6"))
return Distro::RHEL6;
else if (Data.contains("release 5"))
return Distro::RHEL5;
}
return Distro::UnknownDistro;
}
File = VFS.getBufferForFile("/etc/debian_version");
if (File) {
StringRef Data = File.get()->getBuffer();
int MajorVersion;
if (!Data.split('.').first.getAsInteger(10, MajorVersion)) {
switch (MajorVersion) {
case 5:
return Distro::DebianLenny;
case 6:
return Distro::DebianSqueeze;
case 7:
return Distro::DebianWheezy;
case 8:
return Distro::DebianJessie;
case 9:
return Distro::DebianStretch;
case 10:
return Distro::DebianBuster;
case 11:
return Distro::DebianBullseye;
case 12:
return Distro::DebianBookworm;
case 13:
return Distro::DebianTrixie;
default:
return Distro::UnknownDistro;
}
}
return llvm::StringSwitch<Distro::DistroType>(Data.split("\n").first)
.Case("squeeze/sid", Distro::DebianSqueeze)
.Case("wheezy/sid", Distro::DebianWheezy)
.Case("jessie/sid", Distro::DebianJessie)
.Case("stretch/sid", Distro::DebianStretch)
.Case("buster/sid", Distro::DebianBuster)
.Case("bullseye/sid", Distro::DebianBullseye)
.Case("bookworm/sid", Distro::DebianBookworm)
.Default(Distro::UnknownDistro);
}
File = VFS.getBufferForFile("/etc/SuSE-release");
if (File) {
StringRef Data = File.get()->getBuffer();
SmallVector<StringRef, 8> Lines;
Data.split(Lines, "\n");
for (const StringRef &Line : Lines) {
if (!Line.trim().startswith("VERSION"))
continue;
std::pair<StringRef, StringRef> SplitLine = Line.split('=');
std::pair<StringRef, StringRef> SplitVer =
SplitLine.second.trim().split('.');
int Version;
if (!SplitVer.first.getAsInteger(10, Version) && Version > 10)
return Distro::OpenSUSE;
return Distro::UnknownDistro;
}
return Distro::UnknownDistro;
}
if (VFS.exists("/etc/gentoo-release"))
return Distro::Gentoo;
return Distro::UnknownDistro;
}
static Distro::DistroType GetDistro(llvm::vfs::FileSystem &VFS,
const llvm::Triple &TargetOrHost) {
if (!TargetOrHost.isOSLinux())
return Distro::UnknownDistro;
const bool onRealFS = (llvm::vfs::getRealFileSystem() == &VFS);
llvm::Triple HostTriple(llvm::sys::getProcessTriple());
if (!HostTriple.isOSLinux() && onRealFS)
return Distro::UnknownDistro;
if (onRealFS) {
static Distro::DistroType LinuxDistro = DetectDistro(VFS);
return LinuxDistro;
}
return DetectDistro(VFS);
}
Distro::Distro(llvm::vfs::FileSystem &VFS, const llvm::Triple &TargetOrHost)
: DistroVal(GetDistro(VFS, TargetOrHost)) {}