|  | // 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 <stddef.h> | 
|  | #include <stdint.h> | 
|  |  | 
|  | #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<Environment> 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<ChromeOSVersionInfo>::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 | 
|  | std::string SysInfo::GetStrippedReleaseBoard() { | 
|  | std::string board = GetLsbReleaseBoard(); | 
|  | const size_t index = board.find("-signed-"); | 
|  | if (index != std::string::npos) | 
|  | board.resize(index); | 
|  |  | 
|  | return base::ToLowerASCII(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<Environment> 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 |