1094332d3Sopenharmony_ci/* 2094332d3Sopenharmony_ci * Copyright (c) 2022 Huawei Device Co., Ltd. 3094332d3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 4094332d3Sopenharmony_ci * you may not use this file except in compliance with the License. 5094332d3Sopenharmony_ci * You may obtain a copy of the License at 6094332d3Sopenharmony_ci * 7094332d3Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 8094332d3Sopenharmony_ci * 9094332d3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software 10094332d3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS, 11094332d3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12094332d3Sopenharmony_ci * See the License for the specific language governing permissions and 13094332d3Sopenharmony_ci * limitations under the License. 14094332d3Sopenharmony_ci */ 15094332d3Sopenharmony_ci 16094332d3Sopenharmony_ci#include "battery_thread.h" 17094332d3Sopenharmony_ci#include <cerrno> 18094332d3Sopenharmony_ci#include <regex> 19094332d3Sopenharmony_ci#include <sys/epoll.h> 20094332d3Sopenharmony_ci#include <sys/socket.h> 21094332d3Sopenharmony_ci#include <sys/timerfd.h> 22094332d3Sopenharmony_ci#include <unistd.h> 23094332d3Sopenharmony_ci#include <linux/netlink.h> 24094332d3Sopenharmony_ci#include "battery_config.h" 25094332d3Sopenharmony_ci#include "battery_log.h" 26094332d3Sopenharmony_ci 27094332d3Sopenharmony_cinamespace OHOS { 28094332d3Sopenharmony_cinamespace HDI { 29094332d3Sopenharmony_cinamespace Battery { 30094332d3Sopenharmony_cinamespace V2_0 { 31094332d3Sopenharmony_cinamespace { 32094332d3Sopenharmony_ciconstexpr int32_t UEVENT_BUFF_SIZE = (64 * 1024); 33094332d3Sopenharmony_ciconstexpr int32_t UEVENT_RESERVED_SIZE = 2; 34094332d3Sopenharmony_ciconstexpr int32_t UEVENT_MSG_LEN = (2 * 1024); 35094332d3Sopenharmony_ciconstexpr int32_t TIMER_FAST_SEC = 2; 36094332d3Sopenharmony_ciconstexpr int32_t SEC_TO_MSEC = 1000; 37094332d3Sopenharmony_ciconst std::string POWER_SUPPLY = "SUBSYSTEM=power_supply"; 38094332d3Sopenharmony_ci} 39094332d3Sopenharmony_cistatic sptr<IBatteryCallback> g_callback; 40094332d3Sopenharmony_ci 41094332d3Sopenharmony_ciBatteryThread::~BatteryThread() 42094332d3Sopenharmony_ci{ 43094332d3Sopenharmony_ci BATTERY_HILOGW(COMP_HDI, "enter %{public}s", __func__); 44094332d3Sopenharmony_ci isRunning_ = false; 45094332d3Sopenharmony_ci if (batteryThread_ != nullptr && batteryThread_->joinable()) { 46094332d3Sopenharmony_ci batteryThread_->join(); 47094332d3Sopenharmony_ci } 48094332d3Sopenharmony_ci} 49094332d3Sopenharmony_ci 50094332d3Sopenharmony_civoid BatteryThread::InitCallback(const sptr<IBatteryCallback>& callback) 51094332d3Sopenharmony_ci{ 52094332d3Sopenharmony_ci g_callback = callback; 53094332d3Sopenharmony_ci} 54094332d3Sopenharmony_ci 55094332d3Sopenharmony_ciint32_t BatteryThread::OpenUeventSocket() 56094332d3Sopenharmony_ci{ 57094332d3Sopenharmony_ci int32_t bufferSize = UEVENT_BUFF_SIZE; 58094332d3Sopenharmony_ci struct sockaddr_nl address = { 59094332d3Sopenharmony_ci .nl_family = AF_NETLINK, 60094332d3Sopenharmony_ci .nl_pid = getpid(), 61094332d3Sopenharmony_ci .nl_groups = 0xffffffff 62094332d3Sopenharmony_ci }; 63094332d3Sopenharmony_ci 64094332d3Sopenharmony_ci int32_t fd = socket(PF_NETLINK, SOCK_DGRAM | SOCK_CLOEXEC, NETLINK_KOBJECT_UEVENT); 65094332d3Sopenharmony_ci if (fd == INVALID_FD) { 66094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "open uevent socket failed, fd is invalid"); 67094332d3Sopenharmony_ci return INVALID_FD; 68094332d3Sopenharmony_ci } 69094332d3Sopenharmony_ci 70094332d3Sopenharmony_ci int32_t ret = setsockopt(fd, SOL_SOCKET, SO_RCVBUF, &bufferSize, sizeof(bufferSize)); 71094332d3Sopenharmony_ci if (ret < 0) { 72094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "set socket opt failed, ret: %{public}d", ret); 73094332d3Sopenharmony_ci close(fd); 74094332d3Sopenharmony_ci return INVALID_FD; 75094332d3Sopenharmony_ci } 76094332d3Sopenharmony_ci 77094332d3Sopenharmony_ci ret = bind(fd, reinterpret_cast<const struct sockaddr*>(&address), sizeof(struct sockaddr_nl)); 78094332d3Sopenharmony_ci if (ret < 0) { 79094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "bind socket address failed, ret: %{public}d", ret); 80094332d3Sopenharmony_ci close(fd); 81094332d3Sopenharmony_ci return INVALID_FD; 82094332d3Sopenharmony_ci } 83094332d3Sopenharmony_ci return fd; 84094332d3Sopenharmony_ci} 85094332d3Sopenharmony_ci 86094332d3Sopenharmony_ciint32_t BatteryThread::RegisterCallback(int32_t fd, EventType et) 87094332d3Sopenharmony_ci{ 88094332d3Sopenharmony_ci struct epoll_event ev = {0}; 89094332d3Sopenharmony_ci 90094332d3Sopenharmony_ci ev.events = EPOLLIN; 91094332d3Sopenharmony_ci if (et == EVENT_TIMER_FD) { 92094332d3Sopenharmony_ci ev.events |= EPOLLWAKEUP; 93094332d3Sopenharmony_ci } 94094332d3Sopenharmony_ci 95094332d3Sopenharmony_ci ev.data.ptr = reinterpret_cast<void*>(this); 96094332d3Sopenharmony_ci ev.data.fd = fd; 97094332d3Sopenharmony_ci if (epoll_ctl(epFd_, EPOLL_CTL_ADD, fd, &ev) == -1) { 98094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "epoll_ctl failed, error num =%{public}d", errno); 99094332d3Sopenharmony_ci return HDF_FAILURE; 100094332d3Sopenharmony_ci } 101094332d3Sopenharmony_ci return HDF_SUCCESS; 102094332d3Sopenharmony_ci} 103094332d3Sopenharmony_ci 104094332d3Sopenharmony_civoid BatteryThread::UpdateEpollInterval(const int32_t chargeState) 105094332d3Sopenharmony_ci{ 106094332d3Sopenharmony_ci if ((chargeState != PowerSupplyProvider::CHARGE_STATE_NONE) && 107094332d3Sopenharmony_ci (chargeState != PowerSupplyProvider::CHARGE_STATE_RESERVED)) { 108094332d3Sopenharmony_ci epollInterval_ = TIMER_FAST_SEC * SEC_TO_MSEC; 109094332d3Sopenharmony_ci } else { 110094332d3Sopenharmony_ci epollInterval_ = -1; 111094332d3Sopenharmony_ci } 112094332d3Sopenharmony_ci} 113094332d3Sopenharmony_ci 114094332d3Sopenharmony_ciint32_t BatteryThread::InitUevent() 115094332d3Sopenharmony_ci{ 116094332d3Sopenharmony_ci auto& batteryConfig = BatteryConfig::GetInstance(); 117094332d3Sopenharmony_ci batteryConfig.ParseConfig(); 118094332d3Sopenharmony_ci powerUeventMap_ = batteryConfig.GetUeventList(); 119094332d3Sopenharmony_ci 120094332d3Sopenharmony_ci ueventFd_ = OpenUeventSocket(); 121094332d3Sopenharmony_ci if (ueventFd_ == INVALID_FD) { 122094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "open uevent socket failed, fd is invalid"); 123094332d3Sopenharmony_ci return HDF_ERR_BAD_FD; 124094332d3Sopenharmony_ci } 125094332d3Sopenharmony_ci 126094332d3Sopenharmony_ci fcntl(ueventFd_, F_SETFL, O_NONBLOCK); 127094332d3Sopenharmony_ci callbacks_.insert(std::make_pair(ueventFd_, &BatteryThread::UeventCallback)); 128094332d3Sopenharmony_ci 129094332d3Sopenharmony_ci if (RegisterCallback(ueventFd_, EVENT_UEVENT_FD)) { 130094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "register Uevent event failed"); 131094332d3Sopenharmony_ci return HDF_ERR_BAD_FD; 132094332d3Sopenharmony_ci } 133094332d3Sopenharmony_ci return HDF_SUCCESS; 134094332d3Sopenharmony_ci} 135094332d3Sopenharmony_ci 136094332d3Sopenharmony_ciint32_t BatteryThread::Init([[maybe_unused]] void* service) 137094332d3Sopenharmony_ci{ 138094332d3Sopenharmony_ci provider_ = std::make_unique<PowerSupplyProvider>(); 139094332d3Sopenharmony_ci if (provider_ != nullptr) { 140094332d3Sopenharmony_ci provider_->InitBatteryPath(); 141094332d3Sopenharmony_ci provider_->InitPowerSupplySysfs(); 142094332d3Sopenharmony_ci } 143094332d3Sopenharmony_ci 144094332d3Sopenharmony_ci epFd_ = epoll_create1(EPOLL_CLOEXEC); 145094332d3Sopenharmony_ci if (epFd_ == INVALID_FD) { 146094332d3Sopenharmony_ci BATTERY_HILOGE(COMP_HDI, "epoll create failed, epFd_ is invalid"); 147094332d3Sopenharmony_ci return HDF_ERR_BAD_FD; 148094332d3Sopenharmony_ci } 149094332d3Sopenharmony_ci 150094332d3Sopenharmony_ci InitUevent(); 151094332d3Sopenharmony_ci 152094332d3Sopenharmony_ci return HDF_SUCCESS; 153094332d3Sopenharmony_ci} 154094332d3Sopenharmony_ci 155094332d3Sopenharmony_ciint32_t BatteryThread::UpdateWaitInterval() 156094332d3Sopenharmony_ci{ 157094332d3Sopenharmony_ci return HDF_FAILURE; 158094332d3Sopenharmony_ci} 159094332d3Sopenharmony_ci 160094332d3Sopenharmony_civoid BatteryThread::UeventCallback(void* service) 161094332d3Sopenharmony_ci{ 162094332d3Sopenharmony_ci char msg[UEVENT_MSG_LEN + UEVENT_RESERVED_SIZE] = { 0 }; 163094332d3Sopenharmony_ci 164094332d3Sopenharmony_ci ssize_t len = recv(ueventFd_, msg, UEVENT_MSG_LEN, 0); 165094332d3Sopenharmony_ci if (len < 0 || len >= UEVENT_MSG_LEN) { 166094332d3Sopenharmony_ci BATTERY_HILOGI(COMP_HDI, "recv return msg is invalid, len: %{public}zd", len); 167094332d3Sopenharmony_ci return; 168094332d3Sopenharmony_ci } 169094332d3Sopenharmony_ci 170094332d3Sopenharmony_ci // msg separator 171094332d3Sopenharmony_ci msg[len] = '\0'; 172094332d3Sopenharmony_ci msg[len + 1] = '\0'; 173094332d3Sopenharmony_ci std::string powerUevent; 174094332d3Sopenharmony_ci if (!MatchPowerUevent(msg, powerUevent)) { 175094332d3Sopenharmony_ci return; 176094332d3Sopenharmony_ci } 177094332d3Sopenharmony_ci BATTERY_HILOGD(FEATURE_BATT_INFO, "PowerUevent msg:%{public}s", powerUevent.c_str()); 178094332d3Sopenharmony_ci UpdateBatteryInfo(service, powerUevent); 179094332d3Sopenharmony_ci} 180094332d3Sopenharmony_ci 181094332d3Sopenharmony_civoid BatteryThread::UpdateBatteryInfo(void* service, const std::string& powerUevent) 182094332d3Sopenharmony_ci{ 183094332d3Sopenharmony_ci BatteryInfo event = {}; 184094332d3Sopenharmony_ci std::unique_ptr<BatterydInfo> batteryInfo = std::make_unique<BatterydInfo>(); 185094332d3Sopenharmony_ci if (batteryInfo == nullptr) { 186094332d3Sopenharmony_ci BATTERY_HILOGE(FEATURE_BATT_INFO, "make_unique BatterydInfo error"); 187094332d3Sopenharmony_ci return; 188094332d3Sopenharmony_ci } 189094332d3Sopenharmony_ci 190094332d3Sopenharmony_ci provider_->UpdateInfoByReadSysFile(batteryInfo.get()); 191094332d3Sopenharmony_ci event.capacity = batteryInfo->capacity_; 192094332d3Sopenharmony_ci event.voltage= batteryInfo->voltage_; 193094332d3Sopenharmony_ci event.temperature = batteryInfo->temperature_; 194094332d3Sopenharmony_ci event.healthState = batteryInfo->healthState_; 195094332d3Sopenharmony_ci event.pluggedType = batteryInfo->pluggedType_; 196094332d3Sopenharmony_ci event.pluggedMaxCurrent = batteryInfo->pluggedMaxCurrent_; 197094332d3Sopenharmony_ci event.pluggedMaxVoltage = batteryInfo->pluggedMaxVoltage_; 198094332d3Sopenharmony_ci event.chargeState = batteryInfo->chargeState_; 199094332d3Sopenharmony_ci event.chargeCounter = batteryInfo->chargeCounter_; 200094332d3Sopenharmony_ci event.present = batteryInfo->present_; 201094332d3Sopenharmony_ci event.technology = batteryInfo->technology_; 202094332d3Sopenharmony_ci event.curNow = batteryInfo->curNow_; 203094332d3Sopenharmony_ci event.remainEnergy = batteryInfo->remainEnergy_; 204094332d3Sopenharmony_ci event.totalEnergy = batteryInfo->totalEnergy_; 205094332d3Sopenharmony_ci event.uevent = powerUevent; 206094332d3Sopenharmony_ci 207094332d3Sopenharmony_ci if (g_callback != nullptr) { 208094332d3Sopenharmony_ci g_callback->Update(event); 209094332d3Sopenharmony_ci } else { 210094332d3Sopenharmony_ci BATTERY_HILOGI(FEATURE_BATT_INFO, "g_callback is nullptr"); 211094332d3Sopenharmony_ci } 212094332d3Sopenharmony_ci 213094332d3Sopenharmony_ci BATTERY_HILOGI(COMP_DRV, "battery c=%{public}d, v=%{public}d, c=%{public}d, t=%{public}d, " 214094332d3Sopenharmony_ci "h=%{public}d, pt=%{public}d, cs=%{public}d, pmc=%{public}d, " 215094332d3Sopenharmony_ci "pmv=%{public}d, cc=%{public}d, p=%{public}d, re=%{public}d, te=%{public}d", 216094332d3Sopenharmony_ci event.capacity, event.voltage, event.curNow, event.temperature, event.healthState, 217094332d3Sopenharmony_ci event.pluggedType, event.chargeState, event.pluggedMaxCurrent, event.pluggedMaxVoltage, 218094332d3Sopenharmony_ci event.chargeCounter, event.present, event.remainEnergy, event.totalEnergy); 219094332d3Sopenharmony_ci} 220094332d3Sopenharmony_ci 221094332d3Sopenharmony_cibool BatteryThread::MatchPowerUevent(const char* msg, std::string& powerUevent) 222094332d3Sopenharmony_ci{ 223094332d3Sopenharmony_ci while (*msg) { 224094332d3Sopenharmony_ci if (!strcmp(msg, POWER_SUPPLY.c_str())) { 225094332d3Sopenharmony_ci powerUevent = POWER_SUPPLY; 226094332d3Sopenharmony_ci return true; 227094332d3Sopenharmony_ci } 228094332d3Sopenharmony_ci if (CheckPowerUevent(msg, powerUevent)) { 229094332d3Sopenharmony_ci return true; 230094332d3Sopenharmony_ci } 231094332d3Sopenharmony_ci while (*msg++) {} // move to next 232094332d3Sopenharmony_ci } 233094332d3Sopenharmony_ci 234094332d3Sopenharmony_ci return false; 235094332d3Sopenharmony_ci} 236094332d3Sopenharmony_ci 237094332d3Sopenharmony_cibool BatteryThread::CheckPowerUevent(const char* msg, std::string& powerUevent) 238094332d3Sopenharmony_ci{ 239094332d3Sopenharmony_ci auto iter = powerUeventMap_.find(msg); 240094332d3Sopenharmony_ci if (iter != powerUeventMap_.end()) { 241094332d3Sopenharmony_ci while (*msg++) {} 242094332d3Sopenharmony_ci for (auto& uevent : iter->second) { 243094332d3Sopenharmony_ci std::regex r(uevent.first); 244094332d3Sopenharmony_ci if (std::regex_match(msg, r)) { 245094332d3Sopenharmony_ci powerUevent = msg; 246094332d3Sopenharmony_ci powerUevent += "$" + uevent.second; 247094332d3Sopenharmony_ci return true; 248094332d3Sopenharmony_ci } 249094332d3Sopenharmony_ci } 250094332d3Sopenharmony_ci } 251094332d3Sopenharmony_ci return false; 252094332d3Sopenharmony_ci} 253094332d3Sopenharmony_ci 254094332d3Sopenharmony_civoid BatteryThread::LoopingThreadEntry(void* arg) 255094332d3Sopenharmony_ci{ 256094332d3Sopenharmony_ci int32_t nevents = 0; 257094332d3Sopenharmony_ci size_t size = callbacks_.size(); 258094332d3Sopenharmony_ci struct epoll_event events[size]; 259094332d3Sopenharmony_ci 260094332d3Sopenharmony_ci while (isRunning_) { 261094332d3Sopenharmony_ci if (!nevents) { 262094332d3Sopenharmony_ci CycleMatters(); 263094332d3Sopenharmony_ci } 264094332d3Sopenharmony_ci 265094332d3Sopenharmony_ci HandleStates(); 266094332d3Sopenharmony_ci 267094332d3Sopenharmony_ci int32_t timeout = epollInterval_; 268094332d3Sopenharmony_ci int32_t waitTimeout = UpdateWaitInterval(); 269094332d3Sopenharmony_ci if ((timeout < 0) || (waitTimeout > 0 && waitTimeout < timeout)) { 270094332d3Sopenharmony_ci timeout = waitTimeout; 271094332d3Sopenharmony_ci } 272094332d3Sopenharmony_ci 273094332d3Sopenharmony_ci nevents = epoll_wait(epFd_, events, static_cast<int32_t>(size), timeout); 274094332d3Sopenharmony_ci if (nevents <= 0) { 275094332d3Sopenharmony_ci continue; 276094332d3Sopenharmony_ci } 277094332d3Sopenharmony_ci 278094332d3Sopenharmony_ci for (int32_t n = 0; n < nevents; ++n) { 279094332d3Sopenharmony_ci if (events[n].data.ptr) { 280094332d3Sopenharmony_ci auto* func = const_cast<BatteryThread*>(this); 281094332d3Sopenharmony_ci (callbacks_.find(events[n].data.fd)->second)(func, arg); 282094332d3Sopenharmony_ci } 283094332d3Sopenharmony_ci } 284094332d3Sopenharmony_ci } 285094332d3Sopenharmony_ci} 286094332d3Sopenharmony_ci 287094332d3Sopenharmony_civoid BatteryThread::StartThread(void* service) 288094332d3Sopenharmony_ci{ 289094332d3Sopenharmony_ci Init(service); 290094332d3Sopenharmony_ci Run(service); 291094332d3Sopenharmony_ci} 292094332d3Sopenharmony_ci 293094332d3Sopenharmony_civoid BatteryThread::Run(void* service) 294094332d3Sopenharmony_ci{ 295094332d3Sopenharmony_ci batteryThread_ = std::make_unique<std::thread>([this, service] { this->LoopingThreadEntry(service); }); 296094332d3Sopenharmony_ci pthread_setname_np(batteryThread_->native_handle(), "battery_thread"); 297094332d3Sopenharmony_ci} 298094332d3Sopenharmony_ci} // namespace V2_0 299094332d3Sopenharmony_ci} // namespace Battery 300094332d3Sopenharmony_ci} // namespace HDI 301094332d3Sopenharmony_ci} // namespace OHOS 302