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