// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/sys_info.h" #include #include #include "base/environment.h" #include "base/files/file.h" #include "base/files/file_path.h" #include "base/files/file_util.h" #include "base/lazy_instance.h" #include "base/macros.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_piece.h" #include "base/strings/string_split.h" #include "base/strings/string_tokenizer.h" #include "base/strings/string_util.h" #include "base/threading/thread_restrictions.h" namespace base { namespace { const char* const kLinuxStandardBaseVersionKeys[] = { "CHROMEOS_RELEASE_VERSION", "GOOGLE_RELEASE", "DISTRIB_RELEASE", }; const char kChromeOsReleaseNameKey[] = "CHROMEOS_RELEASE_NAME"; const char* const kChromeOsReleaseNames[] = { "Chrome OS", "Chromium OS", }; const char kLinuxStandardBaseReleaseFile[] = "/etc/lsb-release"; const char kLsbReleaseKey[] = "LSB_RELEASE"; const char kLsbReleaseTimeKey[] = "LSB_RELEASE_TIME"; // Seconds since epoch const char kLsbReleaseSourceKey[] = "lsb-release"; const char kLsbReleaseSourceEnv[] = "env"; const char kLsbReleaseSourceFile[] = "file"; class ChromeOSVersionInfo { public: ChromeOSVersionInfo() { Parse(); } void Parse() { lsb_release_map_.clear(); major_version_ = 0; minor_version_ = 0; bugfix_version_ = 0; is_running_on_chromeos_ = false; std::string lsb_release, lsb_release_time_str; std::unique_ptr env(Environment::Create()); bool parsed_from_env = env->GetVar(kLsbReleaseKey, &lsb_release) && env->GetVar(kLsbReleaseTimeKey, &lsb_release_time_str); if (parsed_from_env) { double us = 0; if (StringToDouble(lsb_release_time_str, &us)) lsb_release_time_ = Time::FromDoubleT(us); } else { // If the LSB_RELEASE and LSB_RELEASE_TIME environment variables are not // set, fall back to a blocking read of the lsb_release file. This should // only happen in non Chrome OS environments. ThreadRestrictions::ScopedAllowIO allow_io; FilePath path(kLinuxStandardBaseReleaseFile); ReadFileToString(path, &lsb_release); File::Info fileinfo; if (GetFileInfo(path, &fileinfo)) lsb_release_time_ = fileinfo.creation_time; } ParseLsbRelease(lsb_release); // For debugging: lsb_release_map_[kLsbReleaseSourceKey] = parsed_from_env ? kLsbReleaseSourceEnv : kLsbReleaseSourceFile; } bool GetLsbReleaseValue(const std::string& key, std::string* value) { SysInfo::LsbReleaseMap::const_iterator iter = lsb_release_map_.find(key); if (iter == lsb_release_map_.end()) return false; *value = iter->second; return true; } void GetVersionNumbers(int32_t* major_version, int32_t* minor_version, int32_t* bugfix_version) { *major_version = major_version_; *minor_version = minor_version_; *bugfix_version = bugfix_version_; } const Time& lsb_release_time() const { return lsb_release_time_; } const SysInfo::LsbReleaseMap& lsb_release_map() const { return lsb_release_map_; } bool is_running_on_chromeos() const { return is_running_on_chromeos_; } private: void ParseLsbRelease(const std::string& lsb_release) { // Parse and cache lsb_release key pairs. There should only be a handful // of entries so the overhead for this will be small, and it can be // useful for debugging. base::StringPairs pairs; SplitStringIntoKeyValuePairs(lsb_release, '=', '\n', &pairs); for (size_t i = 0; i < pairs.size(); ++i) { std::string key, value; TrimWhitespaceASCII(pairs[i].first, TRIM_ALL, &key); TrimWhitespaceASCII(pairs[i].second, TRIM_ALL, &value); if (key.empty()) continue; lsb_release_map_[key] = value; } // Parse the version from the first matching recognized version key. std::string version; for (size_t i = 0; i < arraysize(kLinuxStandardBaseVersionKeys); ++i) { std::string key = kLinuxStandardBaseVersionKeys[i]; if (GetLsbReleaseValue(key, &version) && !version.empty()) break; } StringTokenizer tokenizer(version, "."); if (tokenizer.GetNext()) { StringToInt(tokenizer.token_piece(), &major_version_); } if (tokenizer.GetNext()) { StringToInt(tokenizer.token_piece(), &minor_version_); } if (tokenizer.GetNext()) { StringToInt(tokenizer.token_piece(), &bugfix_version_); } // Check release name for Chrome OS. std::string release_name; if (GetLsbReleaseValue(kChromeOsReleaseNameKey, &release_name)) { for (size_t i = 0; i < arraysize(kChromeOsReleaseNames); ++i) { if (release_name == kChromeOsReleaseNames[i]) { is_running_on_chromeos_ = true; break; } } } } Time lsb_release_time_; SysInfo::LsbReleaseMap lsb_release_map_; int32_t major_version_; int32_t minor_version_; int32_t bugfix_version_; bool is_running_on_chromeos_; }; static LazyInstance::Leaky g_chrome_os_version_info = LAZY_INSTANCE_INITIALIZER; ChromeOSVersionInfo& GetChromeOSVersionInfo() { return g_chrome_os_version_info.Get(); } } // namespace // static void SysInfo::OperatingSystemVersionNumbers(int32_t* major_version, int32_t* minor_version, int32_t* bugfix_version) { return GetChromeOSVersionInfo().GetVersionNumbers( major_version, minor_version, bugfix_version); } // static const SysInfo::LsbReleaseMap& SysInfo::GetLsbReleaseMap() { return GetChromeOSVersionInfo().lsb_release_map(); } // static bool SysInfo::GetLsbReleaseValue(const std::string& key, std::string* value) { return GetChromeOSVersionInfo().GetLsbReleaseValue(key, value); } // static std::string SysInfo::GetLsbReleaseBoard() { const char kMachineInfoBoard[] = "CHROMEOS_RELEASE_BOARD"; std::string board; if (!GetLsbReleaseValue(kMachineInfoBoard, &board)) board = "unknown"; return board; } // static Time SysInfo::GetLsbReleaseTime() { return GetChromeOSVersionInfo().lsb_release_time(); } // static bool SysInfo::IsRunningOnChromeOS() { return GetChromeOSVersionInfo().is_running_on_chromeos(); } // static void SysInfo::SetChromeOSVersionInfoForTest(const std::string& lsb_release, const Time& lsb_release_time) { std::unique_ptr env(Environment::Create()); env->SetVar(kLsbReleaseKey, lsb_release); env->SetVar(kLsbReleaseTimeKey, NumberToString(lsb_release_time.ToDoubleT())); g_chrome_os_version_info.Get().Parse(); } } // namespace base