refactor(battery): Cleanup

This commit is contained in:
Michael Carlberg 2016-10-30 14:41:06 +01:00
parent 2f7ec4ceee
commit 36d4ee0e67
2 changed files with 104 additions and 77 deletions

View File

@ -13,6 +13,7 @@
LEMONBUDDY_NS LEMONBUDDY_NS
namespace modules { namespace modules {
enum class battery_state { NONE = 0, UNKNOWN, CHARGING, DISCHARGING, FULL };
class battery_module : public inotify_module<battery_module> { class battery_module : public inotify_module<battery_module> {
public: public:
using inotify_module::inotify_module; using inotify_module::inotify_module;
@ -27,9 +28,6 @@ namespace modules {
m_path_capacity = string_util::replace(PATH_BATTERY_CAPACITY, "%battery%", m_battery); m_path_capacity = string_util::replace(PATH_BATTERY_CAPACITY, "%battery%", m_battery);
m_path_adapter = string_util::replace(PATH_ADAPTER_STATUS, "%adapter%", m_adapter); m_path_adapter = string_util::replace(PATH_ADAPTER_STATUS, "%adapter%", m_adapter);
m_state = STATE_UNKNOWN;
m_percentage = 0;
// }}} // }}}
// Validate paths {{{ // Validate paths {{{
@ -38,6 +36,12 @@ namespace modules {
if (!file_util::exists(m_path_adapter)) if (!file_util::exists(m_path_adapter))
throw module_error("The file '" + m_path_adapter + "' does not exist"); throw module_error("The file '" + m_path_adapter + "' does not exist");
// }}}
// Load state and capacity level {{{
m_percentage = current_percentage();
m_state = current_state();
// }}} // }}}
// Add formats and elements {{{ // Add formats and elements {{{
@ -75,74 +79,55 @@ namespace modules {
} }
void start() { void start() {
m_threads.emplace_back(thread(&battery_module::subthread_routine, this)); m_threads.emplace_back(thread(&battery_module::subthread, this));
inotify_module::start(); inotify_module::start();
} }
void teardown() {
wakeup();
}
bool on_event(inotify_event* event) { bool on_event(inotify_event* event) {
if (event != nullptr) if (event != nullptr) {
m_log.trace("%s: %s", name(), event->filename); m_log.trace("%s: %s", name(), event->filename);
m_notified.store(true, std::memory_order_relaxed);
auto status = file_util::get_contents(m_path_adapter);
if (status.empty()) {
m_log.err("%s: Failed to read '%s'", name(), m_path_adapter);
return false;
} }
auto capacity = file_util::get_contents(m_path_capacity); auto state = current_state();
if (capacity.empty()) { int percentage = m_percentage;
m_log.err("%s: Failed to read '%s'", name(), m_path_capacity);
return false; if (state != battery_state::FULL) {
percentage = current_percentage();
} }
int percentage = math_util::cap<float>(std::atof(capacity.c_str()), 0, 100) + 0.5; // Ignore unchanged state
int state = STATE_UNKNOWN;
switch (status[0]) {
case '0':
state = STATE_DISCHARGING;
break;
case '1':
state = STATE_CHARGING;
break;
}
if (state == STATE_CHARGING) {
if (percentage >= m_fullat)
percentage = 100;
if (percentage == 100)
state = STATE_FULL;
}
// check for nullptr since we don't want to ignore the update for the warmup run
if (event != nullptr && m_state == state && m_percentage == percentage) { if (event != nullptr && m_state == state && m_percentage == percentage) {
m_log.trace("%s: Ignore update since values are unchanged", name());
return false; return false;
} }
m_percentage = percentage;
m_state = state;
if (m_label_charging) { if (m_label_charging) {
m_label_charging->reset_tokens(); m_label_charging->reset_tokens();
m_label_charging->replace_token("%percentage%", to_string(percentage) + "%"); m_label_charging->replace_token("%percentage%", to_string(m_percentage) + "%");
} }
if (m_label_discharging) { if (m_label_discharging) {
m_label_discharging->reset_tokens(); m_label_discharging->reset_tokens();
m_label_discharging->replace_token("%percentage%", to_string(percentage) + "%"); m_label_discharging->replace_token("%percentage%", to_string(m_percentage) + "%");
} }
if (m_label_full) { if (m_label_full) {
m_label_full->reset_tokens(); m_label_full->reset_tokens();
m_label_full->replace_token("%percentage%", to_string(percentage) + "%"); m_label_full->replace_token("%percentage%", to_string(m_percentage) + "%");
} }
m_state = state;
m_percentage = percentage;
return true; return true;
} }
string get_format() const { string get_format() const {
if (m_state == STATE_FULL) if (m_state == battery_state::FULL)
return FORMAT_FULL; return FORMAT_FULL;
else if (m_state == STATE_CHARGING) else if (m_state == battery_state::CHARGING)
return FORMAT_CHARGING; return FORMAT_CHARGING;
else else
return FORMAT_DISCHARGING; return FORMAT_DISCHARGING;
@ -167,42 +152,76 @@ namespace modules {
} }
protected: protected:
void subthread_routine() { /**
this_thread::yield(); * Read the current adapter state from
*/
battery_state current_state() {
auto adapter_status = file_util::get_contents(m_path_adapter);
if (adapter_status.empty()) {
return battery_state::UNKNOWN;
} else if (adapter_status[0] == '0') {
return battery_state::DISCHARGING;
} else if (adapter_status[0] != '1') {
return battery_state::UNKNOWN;
} else if (m_percentage < m_fullat) {
return battery_state::CHARGING;
} else {
return battery_state::FULL;
}
}
/**
* Get the current capacity level
*/
int current_percentage() {
auto capacity = file_util::get_contents(m_path_capacity);
auto value = math_util::cap<int>(std::atof(capacity.c_str()), 0, 100);
if (value >= m_fullat) {
return 100;
} else {
return value;
}
}
/**
* Subthread runner that emit update events
* to refresh <animation-charging> in case it is used.
*
* Will also poll for events as fallback for systems that
* doesn't report inotify events for files on sysfs
*/
void subthread() {
chrono::duration<double> dur = 1s; chrono::duration<double> dur = 1s;
if (m_animation_charging) if (m_animation_charging) {
dur = chrono::duration<double>(float(m_animation_charging->framerate()) / 1000.0f); dur = chrono::duration<double>(float(m_animation_charging->framerate()) / 1000.0f);
int i = 0;
const int poll_seconds = m_conf.get<float>(name(), "poll-interval", 3.0f) / dur.count();
while (running()) {
// TODO(jaagr): Keep track of when the values were last read to determine
// if we need to trigger the event manually or not.
if (poll_seconds > 0 && (++i % poll_seconds) == 0) {
// Trigger an inotify event in case the underlying filesystem doesn't
m_log.trace("%s: Poll battery capacity", name());
file_util::get_contents(m_path_capacity);
i = 0;
}
if (m_state == STATE_CHARGING)
broadcast();
sleep(dur);
} }
m_log.trace("%s: Reached end of battery subthread", name()); const int interval = m_conf.get<float>(name(), "poll-interval", 3.0f) / dur.count();
while (running()) {
for (int i = 0; running() && i < interval; ++i) {
if (m_state == battery_state::CHARGING) {
broadcast();
}
sleep(dur);
}
if (!running() || m_state == battery_state::CHARGING) {
continue;
}
if (!m_notified.load(std::memory_order_relaxed)) {
file_util::get_contents(m_path_capacity);
}
}
m_log.trace("%s: End of subthread", name());
} }
private: private:
static const int STATE_UNKNOWN = 1;
static const int STATE_CHARGING = 2;
static const int STATE_DISCHARGING = 3;
static const int STATE_FULL = 4;
static constexpr auto FORMAT_CHARGING = "format-charging"; static constexpr auto FORMAT_CHARGING = "format-charging";
static constexpr auto FORMAT_DISCHARGING = "format-discharging"; static constexpr auto FORMAT_DISCHARGING = "format-discharging";
static constexpr auto FORMAT_FULL = "format-full"; static constexpr auto FORMAT_FULL = "format-full";
@ -226,9 +245,12 @@ namespace modules {
string m_path_capacity; string m_path_capacity;
string m_path_adapter; string m_path_adapter;
int m_state; battery_state m_state = battery_state::UNKNOWN;
int m_percentage; std::atomic_int m_percentage{0};
int m_fullat;
stateflag m_notified{false};
int m_fullat = 100;
}; };
} }

View File

@ -176,11 +176,9 @@ namespace modules {
, m_builder(make_unique<builder>(bar)) , m_builder(make_unique<builder>(bar))
, m_formatter(make_unique<module_formatter>(m_conf, m_name)) {} , m_formatter(make_unique<module_formatter>(m_conf, m_name)) {}
~module() { ~module() noexcept {
m_log.trace("%s: Deconstructing", name()); m_log.trace("%s: Deconstructing", name());
assert(!running());
for (auto&& thread_ : m_threads) { for (auto&& thread_ : m_threads) {
if (thread_.joinable()) { if (thread_.joinable()) {
thread_.join(); thread_.join();
@ -511,11 +509,18 @@ namespace modules {
if (w->poll(1000 / watches.size())) { if (w->poll(1000 / watches.size())) {
auto event = w->get_event(); auto event = w->get_event();
w->remove(); for (auto&& w : watches) {
try {
w->remove();
} catch (const system_error&) {
}
}
if (CAST_MOD(Impl)->on_event(event.get())) if (CAST_MOD(Impl)->on_event(event.get()))
CAST_MOD(Impl)->broadcast(); CAST_MOD(Impl)->broadcast();
CAST_MOD(Impl)->idle();
return; return;
} }