/*
* Copyright (C) 2018 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.
*/
#include "CycleCountBackupRestore.h"
namespace device {
namespace google {
namespace marlin {
namespace health {
static constexpr char kCycCntFile[] = "sys/class/power_supply/bms/device/cycle_counts_bins";
static constexpr char kSysPersistFile[] = "/persist/battery/qcom_cycle_counts_bins";
static constexpr int kBuffSize = 256;
CycleCountBackupRestore::CycleCountBackupRestore() { }
void CycleCountBackupRestore::Restore()
{
ReadFromStorage();
ReadFromSRAM();
UpdateAndSave();
}
void CycleCountBackupRestore::Backup()
{
ReadFromSRAM();
UpdateAndSave();
}
void CycleCountBackupRestore::ReadFromStorage()
{
std::string buffer;
if (!android::base::ReadFileToString(std::string(kSysPersistFile), &buffer)) {
LOG(ERROR) << "Cannot read the storage file";
return;
}
if (sscanf(buffer.c_str(), "%d %d %d %d %d %d %d %d",
&sw_bins_[0], &sw_bins_[1], &sw_bins_[2], &sw_bins_[3],
&sw_bins_[4], &sw_bins_[5], &sw_bins_[6], &sw_bins_[7])
!= kBucketCount)
LOG(ERROR) << "data format is wrong in the storage file: " << buffer;
else
LOG(INFO) << "Storage data: " << buffer;
}
void CycleCountBackupRestore::SaveToStorage()
{
char strData[kBuffSize];
snprintf(strData, kBuffSize, "%d %d %d %d %d %d %d %d",
sw_bins_[0], sw_bins_[1], sw_bins_[2], sw_bins_[3],
sw_bins_[4], sw_bins_[5], sw_bins_[6], sw_bins_[7]);
LOG(INFO) << "Save to Storage: " << strData;
if (!android::base::WriteStringToFile(strData, std::string(kSysPersistFile)))
LOG(ERROR) << "Write file error: " << strerror(errno);
}
void CycleCountBackupRestore::ReadFromSRAM()
{
std::string buffer;
if (!android::base::ReadFileToString(std::string(kCycCntFile), &buffer)) {
LOG(ERROR) << "Read cycle counter error: " << strerror(errno);
return;
}
buffer = android::base::Trim(buffer);
if (sscanf(buffer.c_str(), "%d %d %d %d %d %d %d %d",
&hw_bins_[0], &hw_bins_[1], &hw_bins_[2], &hw_bins_[3],
&hw_bins_[4], &hw_bins_[5], &hw_bins_[6], &hw_bins_[7])
!= kBucketCount)
LOG(ERROR) << "Failed to parse SRAM bins: " << buffer;
else
LOG(INFO) << "SRAM data: " << buffer;
}
void CycleCountBackupRestore::SaveToSRAM()
{
char strData[kBuffSize];
snprintf(strData, kBuffSize, "%d %d %d %d %d %d %d %d",
hw_bins_[0], hw_bins_[1], hw_bins_[2], hw_bins_[3],
hw_bins_[4], hw_bins_[5], hw_bins_[6], hw_bins_[7]);
LOG(INFO) << "Save to SRAM: " << strData ;
if (!android::base::WriteStringToFile(strData, std::string(kCycCntFile)))
LOG(ERROR) << "Write data error: " << strerror(errno);
}
void CycleCountBackupRestore::UpdateAndSave()
{
bool backup = false;
bool restore = false;
for (int i = 0; i < kBucketCount; i++) {
if (hw_bins_[i] < sw_bins_[i]) {
hw_bins_[i] = sw_bins_[i];
restore = true;
} else if (hw_bins_[i] > sw_bins_[i]) {
sw_bins_[i] = hw_bins_[i];
backup = true;
}
}
if (restore)
SaveToSRAM();
if (backup)
SaveToStorage();
}
} // namespace health
} // namespace marlin
} // namespace google
} // namespace device