/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "storaged" #include <stdio.h> #include <string.h> #include <sys/statvfs.h> #include <numeric> #include <android-base/file.h> #include <android-base/parseint.h> #include <android-base/logging.h> #include <android-base/strings.h> #include <log/log_event_list.h> #include "storaged.h" #include "storaged_info.h" using namespace std; using namespace chrono; using namespace android::base; using namespace storaged_proto; using android::hardware::health::V2_0::IHealth; using android::hardware::health::V2_0::Result; using android::hardware::health::V2_0::StorageInfo; const string emmc_info_t::emmc_sysfs = "/sys/bus/mmc/devices/mmc0:0001/"; const string emmc_info_t::emmc_debugfs = "/d/mmc0/mmc0:0001/ext_csd"; const char* emmc_info_t::emmc_ver_str[9] = { "4.0", "4.1", "4.2", "4.3", "Obsolete", "4.41", "4.5", "5.0", "5.1" }; const string ufs_info_t::health_file = "/sys/devices/soc/624000.ufshc/health"; namespace { bool FileExists(const std::string& filename) { struct stat buffer; return stat(filename.c_str(), &buffer) == 0; } } // namespace storage_info_t* storage_info_t::get_storage_info(const sp<IHealth>& healthService) { if (healthService != nullptr) { return new health_storage_info_t(healthService); } if (FileExists(emmc_info_t::emmc_sysfs) || FileExists(emmc_info_t::emmc_debugfs)) { return new emmc_info_t; } if (FileExists(ufs_info_t::health_file)) { return new ufs_info_t; } return new storage_info_t; } void storage_info_t::load_perf_history_proto(const IOPerfHistory& perf_history) { Mutex::Autolock _l(si_mutex); if (!perf_history.has_day_start_sec() || perf_history.daily_perf_size() > (int)daily_perf.size() || perf_history.weekly_perf_size() > (int)weekly_perf.size()) { LOG_TO(SYSTEM, ERROR) << "Invalid IOPerfHistory proto"; return; } day_start_tp = {}; day_start_tp += chrono::seconds(perf_history.day_start_sec()); nr_samples = perf_history.nr_samples(); if (nr_samples < recent_perf.size()) { recent_perf.erase(recent_perf.begin() + nr_samples, recent_perf.end()); } size_t i = 0; for (auto bw : perf_history.recent_perf()) { if (i < recent_perf.size()) { recent_perf[i] = bw; } else { recent_perf.push_back(bw); } ++i; } nr_days = perf_history.nr_days(); i = 0; for (auto bw : perf_history.daily_perf()) { daily_perf[i++] = bw; } nr_weeks = perf_history.nr_weeks(); i = 0; for (auto bw : perf_history.weekly_perf()) { weekly_perf[i++] = bw; } } void storage_info_t::refresh(IOPerfHistory* perf_history) { struct statvfs buf; if (statvfs(userdata_path.c_str(), &buf) != 0) { PLOG_TO(SYSTEM, WARNING) << "Failed to get userdata info"; return; } userdata_total_kb = buf.f_bsize * buf.f_blocks >> 10; userdata_free_kb = buf.f_bfree * buf.f_blocks >> 10; Mutex::Autolock _l(si_mutex); perf_history->Clear(); perf_history->set_day_start_sec( duration_cast<chrono::seconds>(day_start_tp.time_since_epoch()).count()); for (const uint32_t& bw : recent_perf) { perf_history->add_recent_perf(bw); } perf_history->set_nr_samples(nr_samples); for (const uint32_t& bw : daily_perf) { perf_history->add_daily_perf(bw); } perf_history->set_nr_days(nr_days); for (const uint32_t& bw : weekly_perf) { perf_history->add_weekly_perf(bw); } perf_history->set_nr_weeks(nr_weeks); } void storage_info_t::publish() { android_log_event_list(EVENTLOGTAG_EMMCINFO) << version << eol << lifetime_a << lifetime_b << LOG_ID_EVENTS; } void storage_info_t::update_perf_history(uint32_t bw, const time_point<system_clock>& tp) { Mutex::Autolock _l(si_mutex); if (tp > day_start_tp && duration_cast<chrono::seconds>(tp - day_start_tp).count() < DAY_TO_SEC) { if (nr_samples >= recent_perf.size()) { recent_perf.push_back(bw); } else { recent_perf[nr_samples] = bw; } nr_samples++; return; } if (nr_samples < recent_perf.size()) { recent_perf.erase(recent_perf.begin() + nr_samples, recent_perf.end()); } uint32_t daily_avg_bw = 0; if (!recent_perf.empty()) { daily_avg_bw = accumulate(recent_perf.begin(), recent_perf.end(), 0) / recent_perf.size(); } day_start_tp = tp - chrono::seconds(duration_cast<chrono::seconds>( tp.time_since_epoch()).count() % DAY_TO_SEC); nr_samples = 0; if (recent_perf.empty()) recent_perf.resize(1); recent_perf[nr_samples++] = bw; if (nr_days < WEEK_TO_DAYS) { daily_perf[nr_days++] = daily_avg_bw; return; } DCHECK(nr_days > 0); uint32_t week_avg_bw = accumulate(daily_perf.begin(), daily_perf.begin() + nr_days, 0) / nr_days; nr_days = 0; daily_perf[nr_days++] = daily_avg_bw; if (nr_weeks >= YEAR_TO_WEEKS) { nr_weeks = 0; } weekly_perf[nr_weeks++] = week_avg_bw; } vector<int> storage_info_t::get_perf_history() { Mutex::Autolock _l(si_mutex); vector<int> ret(3 + recent_perf.size() + daily_perf.size() + weekly_perf.size()); ret[0] = recent_perf.size(); ret[1] = daily_perf.size(); ret[2] = weekly_perf.size(); int start = 3; for (size_t i = 0; i < recent_perf.size(); i++) { int idx = (recent_perf.size() + nr_samples - 1 - i) % recent_perf.size(); ret[start + i] = recent_perf[idx]; } start += recent_perf.size(); for (size_t i = 0; i < daily_perf.size(); i++) { int idx = (daily_perf.size() + nr_days - 1 - i) % daily_perf.size(); ret[start + i] = daily_perf[idx]; } start += daily_perf.size(); for (size_t i = 0; i < weekly_perf.size(); i++) { int idx = (weekly_perf.size() + nr_weeks - 1 - i) % weekly_perf.size(); ret[start + i] = weekly_perf[idx]; } return ret; } uint32_t storage_info_t::get_recent_perf() { Mutex::Autolock _l(si_mutex); if (recent_perf.size() == 0) return 0; return accumulate(recent_perf.begin(), recent_perf.end(), recent_perf.size() / 2) / recent_perf.size(); } void emmc_info_t::report() { if (!report_sysfs() && !report_debugfs()) return; publish(); } bool emmc_info_t::report_sysfs() { string buffer; uint16_t rev = 0; if (!ReadFileToString(emmc_sysfs + "rev", &buffer)) { return false; } if (sscanf(buffer.c_str(), "0x%hx", &rev) < 1 || rev < 7 || rev > ARRAY_SIZE(emmc_ver_str)) { return false; } version = "emmc "; version += emmc_ver_str[rev]; if (!ReadFileToString(emmc_sysfs + "pre_eol_info", &buffer)) { return false; } if (sscanf(buffer.c_str(), "%hx", &eol) < 1 || eol == 0) { return false; } if (!ReadFileToString(emmc_sysfs + "life_time", &buffer)) { return false; } if (sscanf(buffer.c_str(), "0x%hx 0x%hx", &lifetime_a, &lifetime_b) < 2 || (lifetime_a == 0 && lifetime_b == 0)) { return false; } return true; } namespace { const size_t EXT_CSD_FILE_MIN_SIZE = 1024; /* 2 characters in string for each byte */ const size_t EXT_CSD_REV_IDX = 192 * 2; const size_t EXT_PRE_EOL_INFO_IDX = 267 * 2; const size_t EXT_DEVICE_LIFE_TIME_EST_A_IDX = 268 * 2; const size_t EXT_DEVICE_LIFE_TIME_EST_B_IDX = 269 * 2; } // namespace bool emmc_info_t::report_debugfs() { string buffer; uint16_t rev = 0; if (!ReadFileToString(emmc_debugfs, &buffer) || buffer.length() < (size_t)EXT_CSD_FILE_MIN_SIZE) { return false; } string str = buffer.substr(EXT_CSD_REV_IDX, 2); if (!ParseUint(str, &rev) || rev < 7 || rev > ARRAY_SIZE(emmc_ver_str)) { return false; } version = "emmc "; version += emmc_ver_str[rev]; str = buffer.substr(EXT_PRE_EOL_INFO_IDX, 2); if (!ParseUint(str, &eol)) { return false; } str = buffer.substr(EXT_DEVICE_LIFE_TIME_EST_A_IDX, 2); if (!ParseUint(str, &lifetime_a)) { return false; } str = buffer.substr(EXT_DEVICE_LIFE_TIME_EST_B_IDX, 2); if (!ParseUint(str, &lifetime_b)) { return false; } return true; } void ufs_info_t::report() { string buffer; if (!ReadFileToString(health_file, &buffer)) { return; } vector<string> lines = Split(buffer, "\n"); if (lines.empty()) { return; } char rev[8]; if (sscanf(lines[0].c_str(), "ufs version: 0x%7s\n", rev) < 1) { return; } version = "ufs " + string(rev); for (size_t i = 1; i < lines.size(); i++) { char token[32]; uint16_t val; int ret; if ((ret = sscanf(lines[i].c_str(), "Health Descriptor[Byte offset 0x%*d]: %31s = 0x%hx", token, &val)) < 2) { continue; } if (string(token) == "bPreEOLInfo") { eol = val; } else if (string(token) == "bDeviceLifeTimeEstA") { lifetime_a = val; } else if (string(token) == "bDeviceLifeTimeEstB") { lifetime_b = val; } } if (eol == 0 || (lifetime_a == 0 && lifetime_b == 0)) { return; } publish(); } void health_storage_info_t::report() { auto ret = mHealth->getStorageInfo([this](auto result, const auto& halInfos) { if (result == Result::NOT_SUPPORTED) { LOG_TO(SYSTEM, DEBUG) << "getStorageInfo is not supported on health HAL."; return; } if (result != Result::SUCCESS || halInfos.size() == 0) { LOG_TO(SYSTEM, ERROR) << "getStorageInfo failed with result " << toString(result) << " and size " << halInfos.size(); return; } set_values_from_hal_storage_info(halInfos[0]); publish(); }); if (!ret.isOk()) { LOG_TO(SYSTEM, ERROR) << "getStorageInfo failed with " << ret.description(); } } void health_storage_info_t::set_values_from_hal_storage_info(const StorageInfo& halInfo) { eol = halInfo.eol; lifetime_a = halInfo.lifetimeA; lifetime_b = halInfo.lifetimeB; version = halInfo.version; }