1 /* 2 * Copyright (c) 2021 Huawei Device Co., Ltd. 3 * Licensed under the Apache License, Version 2.0 (the "License"); 4 * you may not use this file except in compliance with the License. 5 * You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software 10 * distributed under the License is distributed on an "AS IS" BASIS, 11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 * See the License for the specific language governing permissions and 13 * limitations under the License. 14 */ 15 16 #ifndef UTILS_TIMER_H 17 #define UTILS_TIMER_H 18 19 #include <sys/types.h> 20 #include <cstdint> 21 #include <string> 22 #include <list> 23 #include <map> 24 #include <mutex> 25 #include <thread> 26 #include <vector> 27 28 #include "../src/event_reactor.h" 29 30 namespace OHOS { 31 namespace Utils { 32 /* 33 * Notice: 34 * 1. Timer should be set up(via Setup()) before use, and shutdown(via Shutdown()) before its deconstruction. 35 * 36 * 2. Timer should be set up first and then shutdown. 37 * Avoid delegating them to different threads since it may cause multithreading problem. 38 * 39 * 3. Set up Timer again would not reset this Timer, but return `TIMER_ERR_INVALID_VALUE`. 40 * If a reset operation is required, shut Timer down first and then set it up. 41 * 42 * 4. Parameter in Shutdown() determines whether the thread in Timer would be detach or join. 43 * (True(default) --> join; False --> detach). 44 * + Detach operation would cause possible multithreading problems, thus is not recommended. 45 * If a detach operation is required, availability of related objects used in `thread_` should be guaranteed. 46 */ 47 class Timer { 48 public: 49 using TimerCallback = std::function<void ()>; 50 using TimerCallbackPtr = std::shared_ptr<TimerCallback>; 51 using TimerListCallback = std::function<void (int timerFd)>; 52 53 public: 54 /* 55 * if performance-sensitive, change "timeout" larger before Setup 56 * default-value(1000ms), performance-estimate: occupy fixed-100us in every default-value(1000ms) 57 * timeout: range [-1, INT32MAX], but [-1,0] is not recommended 58 * -1: wait for ever(until event-trigger); 59 * 0: no wait, occupy too much cpu time; 60 * others: wait(until event-trigger) 61 */ 62 explicit Timer(const std::string& name, int timeoutMs = 1000); ~Timer()63 virtual ~Timer() {} 64 65 virtual uint32_t Setup(); 66 67 /* 68 * useJoin true: use std::thread::join(default) 69 * false: use std::thread::detach(not recommended) 70 * if timeoutMs = -1 and no valid event-trigger in epoll_wait: 71 * use std::thread::detach inside to avoid deadloop 72 */ 73 virtual void Shutdown(bool useJoin = true); 74 75 uint32_t Register(const TimerCallback& callback, uint32_t interval /* ms */, bool once = false); 76 void Unregister(uint32_t timerId); 77 78 private: 79 void MainLoop(); 80 void OnTimer(int timerFd); 81 virtual uint32_t DoRegister(const TimerListCallback& callback, uint32_t interval, bool once, int &timerFd); 82 virtual void DoUnregister(uint32_t interval); 83 void DoTimerListCallback(const TimerListCallback& callback, int timerFd); 84 uint32_t GetValidId(uint32_t timerId) const; 85 int GetTimerFd(uint32_t interval /* ms */); 86 void EraseUnusedTimerId(uint32_t interval, const std::vector<uint32_t>& unusedIds); 87 88 private: 89 struct TimerEntry { 90 uint32_t timerId; // unique id 91 uint32_t interval; // million second 92 TimerCallback callback; 93 bool once; 94 int timerFd; 95 }; 96 97 using TimerEntryPtr = std::shared_ptr<TimerEntry>; 98 using TimerEntryList = std::list<TimerEntryPtr>; 99 100 std::map<uint32_t, TimerEntryList> intervalToTimers_; // interval to TimerEntryList 101 std::map<uint32_t, TimerEntryPtr> timerToEntries_; // timer_id to TimerEntry 102 103 std::string name_; 104 int timeoutMs_; 105 std::thread thread_; 106 std::unique_ptr<EventReactor> reactor_; 107 std::map<uint32_t, uint32_t> timers_; // timer_fd to interval 108 std::mutex mutex_; 109 }; 110 111 } // namespace Utils 112 } // namespace OHOS 113 #endif 114 115