/*
* 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 <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"
using namespace std;
using namespace android::base;
void report_storage_health()
{
emmc_info_t mmc;
ufs_info_t ufs;
mmc.report();
ufs.report();
}
void storage_info_t::publish()
{
android_log_event_list(EVENTLOGTAG_EMMCINFO)
<< version << eol << lifetime_a << lifetime_b
<< LOG_ID_EVENTS;
}
bool emmc_info_t::report()
{
if (!report_sysfs() && !report_debugfs())
return false;
publish();
return true;
}
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;
}
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;
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;
}
bool ufs_info_t::report()
{
string buffer;
if (!ReadFileToString(health_file, &buffer)) {
return false;
}
vector<string> lines = Split(buffer, "\n");
if (lines.empty()) {
return false;
}
char rev[8];
if (sscanf(lines[0].c_str(), "ufs version: 0x%7s\n", rev) < 1) {
return false;
}
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 false;
}
publish();
return true;
}