/*************************************************************************** * This file is part of awesome-widgets * * * * awesome-widgets is free software: you can redistribute it and/or * * modify it under the terms of the GNU General Public License as * * published by the Free Software Foundation, either version 3 of the * * License, or (at your option) any later version. * * * * awesome-widgets is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with awesome-widgets. If not, see http://www.gnu.org/licenses/ * ***************************************************************************/ #include "batterysource.h" #include #include #include "awdebug.h" BatterySource::BatterySource(QObject *_parent, QString _acpiPath) : AbstractExtSysMonSource(_parent) , m_acpiPath(std::move(_acpiPath)) { qCDebug(LOG_ESS) << __PRETTY_FUNCTION__; auto directory = QDir(m_acpiPath); m_batteriesCount = directory.entryList(QStringList({"BAT*"}), QDir::Dirs | QDir::NoDotAndDotDot).count(); qCInfo(LOG_ESS) << "Init batteries count as" << m_batteriesCount; } QVariant BatterySource::data(const QString &_source) { qCDebug(LOG_ESS) << "Source" << _source; if (!m_values.contains(_source)) run(); // syncronous update of all values return m_values.take(_source); } void BatterySource::run() { // adaptor QFile acFile(QString("%1/AC/online").arg(m_acpiPath)); if (acFile.open(QIODevice::ReadOnly | QIODevice::Text)) m_values["ac"] = (QString(acFile.readLine()).trimmed().toInt() == 1); acFile.close(); // batteries float currentLevel = 0.0, fullLevel = 0.0; for (int i = 0; i < m_batteriesCount; i++) { // current level QFile currentLevelFile(QString("%1/BAT%2/energy_now").arg(m_acpiPath).arg(i)); if (currentLevelFile.open(QIODevice::ReadOnly | QIODevice::Text)) { auto value = QString(currentLevelFile.readLine()).toInt(); m_trend[i + 1].append(value); m_values[QString("batnow%1").arg(i)] = value; } currentLevelFile.close(); // total QFile fullLevelFile(QString("%1/BAT%2/energy_full").arg(m_acpiPath).arg(i)); if (fullLevelFile.open(QIODevice::ReadOnly | QIODevice::Text)) m_values[QString("battotal%1").arg(i)] = QString(fullLevelFile.readLine()).toInt(); fullLevelFile.close(); m_values[QString("bat%1").arg(i)] = static_cast(100 * m_values[QString("batnow%1").arg(i)].toFloat() / m_values[QString("battotal%1").arg(i)].toFloat()); // accumulate currentLevel += m_values[QString("batnow%1").arg(i)].toFloat(); fullLevel += m_values[QString("battotal%1").arg(i)].toFloat(); } // total m_trend[0].append(static_cast(currentLevel)); m_values["batnow"] = static_cast(currentLevel); m_values["battotal"] = static_cast(fullLevel); m_values["bat"] = static_cast(100 * currentLevel / fullLevel); calculateRates(); } QHash BatterySource::sources() const { auto result = QHash(); // fixed fields result.insert("ac", makeSensorInfo("Is AC online or not", QVariant::Bool)); result.insert("bat", makeSensorInfo("Average battery usage", QVariant::Int, KSysGuard::UnitPercent, 0, 100)); result.insert("batleft", makeSensorInfo("Battery discharge time", QVariant::Int, KSysGuard::UnitSecond)); result.insert("batnow", makeSensorInfo("Current battery capacity", QVariant::Int)); result.insert("batrate", makeSensorInfo("Average battery discharge rate", QVariant::Double, KSysGuard::UnitRate)); result.insert("battotal", makeSensorInfo("Full battery capacity", QVariant::Int)); // generators for (auto i = 0; i < m_batteriesCount; i++) { result.insert(QString("bat%1").arg(i), makeSensorInfo(QString("Battery %1 usage").arg(i), QVariant::Int, KSysGuard::UnitPercent, 0, 100)); result.insert(QString("batleft%1").arg(i), makeSensorInfo(QString("Battery %1 discharge time").arg(i), QVariant::Int, KSysGuard::UnitSecond)); result.insert(QString("batnow%1").arg(i), makeSensorInfo(QString("Battery %1 capacity").arg(i), QVariant::Int)); result.insert(QString("batrate%1").arg(i), makeSensorInfo(QString("Battery %1 discharge rate").arg(i), QVariant::Double, KSysGuard::UnitRate)); result.insert(QString("battotal%1").arg(i), makeSensorInfo(QString("Battery %1 full capacity").arg(i), QVariant::Int)); } return result; } double BatterySource::approximate(const QList &_trend) { qCDebug(LOG_ESS) << "Approximate by MSD" << _trend; auto count = _trend.count(); auto sumxy = 0; for (int i = 0; i < count; i++) sumxy += _trend.at(i) * (i + 1); auto sumx = count * (count + 1) / 2; auto sumy = std::accumulate(_trend.cbegin(), _trend.cend(), 0); auto sumx2 = count * (count + 1) * (2 * count + 1) / 6; return (count * sumxy - sumx * sumy) / (count * sumx2 - std::pow(sumx, 2)); } void BatterySource::calculateRates() { // check if it is first run if (!m_timestamp.isValid()) { m_timestamp = QDateTime::currentDateTimeUtc(); return; } // check time interval auto now = QDateTime::currentDateTimeUtc(); auto interval = m_timestamp.secsTo(now); qCDebug(LOG_AW) << interval; m_timestamp.swap(now); for (int i = 0; i < m_batteriesCount; i++) { auto approx = approximate(m_trend[i + 1]); m_values[QString("batrate%1").arg(i)] = approx / interval; m_values[QString("batleft%1").arg(i)] = interval * m_values[QString("batnow%1").arg(i)].toFloat() / approx; } // total auto approx = approximate(m_trend[0]); m_values["batrate"] = approx / interval; m_values["batleft"] = interval * m_values["batnow"].toFloat() / approx; // old data cleanup for (auto &trend : m_trend.keys()) { if (m_trend[trend].count() <= TREND_LIMIT) continue; m_trend[trend].removeFirst(); } }