1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef HIDL_CALLBACK_UTIL_H_
18 #define HIDL_CALLBACK_UTIL_H_
19 
20 #include <set>
21 
22 #include <hidl/HidlSupport.h>
23 
24 namespace {
25 // Type of callback invoked by the death handler.
26 using on_death_cb_function = std::function<void(uint64_t)>;
27 
28 // Private class used to keep track of death of individual
29 // callbacks stored in HidlCallbackHandler.
30 template <typename CallbackType>
31 class HidlDeathHandler : public android::hardware::hidl_death_recipient {
32    public:
HidlDeathHandler(const on_death_cb_function & user_cb_function)33     HidlDeathHandler(const on_death_cb_function& user_cb_function)
34         : cb_function_(user_cb_function) {}
35     ~HidlDeathHandler() = default;
36 
37     // Death notification for callbacks.
serviceDied(uint64_t cookie,const android::wp<android::hidl::base::V1_0::IBase> &)38     void serviceDied(
39         uint64_t cookie,
40         const android::wp<android::hidl::base::V1_0::IBase>& /* who */)
41         override {
42         cb_function_(cookie);
43     }
44 
45    private:
46     on_death_cb_function cb_function_;
47 
48     DISALLOW_COPY_AND_ASSIGN(HidlDeathHandler);
49 };
50 }  // namespace
51 
52 namespace android {
53 namespace hardware {
54 namespace wifi {
55 namespace V1_2 {
56 namespace implementation {
57 namespace hidl_callback_util {
58 template <typename CallbackType>
59 // Provides a class to manage callbacks for the various HIDL interfaces and
60 // handle the death of the process hosting each callback.
61 class HidlCallbackHandler {
62    public:
HidlCallbackHandler()63     HidlCallbackHandler()
64         : death_handler_(new HidlDeathHandler<CallbackType>(
65               std::bind(&HidlCallbackHandler::onObjectDeath, this,
66                         std::placeholders::_1))) {}
67     ~HidlCallbackHandler() = default;
68 
addCallback(const sp<CallbackType> & cb)69     bool addCallback(const sp<CallbackType>& cb) {
70         // TODO(b/33818800): Can't compare proxies yet. So, use the cookie
71         // (callback proxy's raw pointer) to track the death of individual
72         // clients.
73         uint64_t cookie = reinterpret_cast<uint64_t>(cb.get());
74         if (cb_set_.find(cb) != cb_set_.end()) {
75             LOG(WARNING) << "Duplicate death notification registration";
76             return true;
77         }
78         if (!cb->linkToDeath(death_handler_, cookie)) {
79             LOG(ERROR) << "Failed to register death notification";
80             return false;
81         }
82         cb_set_.insert(cb);
83         return true;
84     }
85 
getCallbacks()86     const std::set<android::sp<CallbackType>>& getCallbacks() {
87         return cb_set_;
88     }
89 
90     // Death notification for callbacks.
onObjectDeath(uint64_t cookie)91     void onObjectDeath(uint64_t cookie) {
92         CallbackType* cb = reinterpret_cast<CallbackType*>(cookie);
93         const auto& iter = cb_set_.find(cb);
94         if (iter == cb_set_.end()) {
95             LOG(ERROR) << "Unknown callback death notification received";
96             return;
97         }
98         cb_set_.erase(iter);
99         LOG(DEBUG) << "Dead callback removed from list";
100     }
101 
invalidate()102     void invalidate() {
103         for (const sp<CallbackType>& cb : cb_set_) {
104             if (!cb->unlinkToDeath(death_handler_)) {
105                 LOG(ERROR) << "Failed to deregister death notification";
106             }
107         }
108         cb_set_.clear();
109     }
110 
111    private:
112     std::set<sp<CallbackType>> cb_set_;
113     sp<HidlDeathHandler<CallbackType>> death_handler_;
114 
115     DISALLOW_COPY_AND_ASSIGN(HidlCallbackHandler);
116 };
117 
118 }  // namespace hidl_callback_util
119 }  // namespace implementation
120 }  // namespace V1_2
121 }  // namespace wifi
122 }  // namespace hardware
123 }  // namespace android
124 #endif  // HIDL_CALLBACK_UTIL_H_
125