• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2022-2023 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 #include "dm_discovery_manager.h"
17 #include "dm_discovery_filter.h"
18 #include "dm_anonymous.h"
19 #include "dm_constants.h"
20 #include "dm_log.h"
21 #include "parameter.h"
22 
23 namespace OHOS {
24 namespace DistributedHardware {
25 constexpr const char* DISCOVERY_TIMEOUT_TASK = "deviceManagerTimer:discovery";
26 const int32_t DISCOVERY_TIMEOUT = 120;
27 
DmDiscoveryManager(std::shared_ptr<SoftbusConnector> softbusConnector,std::shared_ptr<IDeviceManagerServiceListener> listener,std::shared_ptr<HiChainConnector> hiChainConnector)28 DmDiscoveryManager::DmDiscoveryManager(std::shared_ptr<SoftbusConnector> softbusConnector,
29                                        std::shared_ptr<IDeviceManagerServiceListener> listener,
30                                        std::shared_ptr<HiChainConnector> hiChainConnector)
31     : softbusConnector_(softbusConnector), listener_(listener), hiChainConnector_(hiChainConnector)
32 {
33     LOGI("DmDiscoveryManager constructor");
34 }
35 
~DmDiscoveryManager()36 DmDiscoveryManager::~DmDiscoveryManager()
37 {
38     LOGI("DmDiscoveryManager destructor");
39 }
40 
CfgDiscoveryTimer()41 void DmDiscoveryManager::CfgDiscoveryTimer()
42 {
43     if (timer_ == nullptr) {
44         timer_ = std::make_shared<DmTimer>();
45     }
46     timer_->StartTimer(std::string(DISCOVERY_TIMEOUT_TASK), DISCOVERY_TIMEOUT,
47         [this] (std::string name) {
48             DmDiscoveryManager::HandleDiscoveryTimeout(name);
49         });
50 }
51 
CheckDiscoveryQueue(const std::string & pkgName)52 int32_t DmDiscoveryManager::CheckDiscoveryQueue(const std::string &pkgName)
53 {
54     uint16_t subscribeId = 0;
55     std::string frontPkgName = "";
56     {
57         std::lock_guard<std::mutex> autoLock(locks_);
58         if (discoveryQueue_.empty()) {
59             return DM_OK;
60         }
61 
62         frontPkgName = discoveryQueue_.front();
63         if (pkgName == frontPkgName) {
64             LOGE("DmDiscoveryManager::StartDeviceDiscovery repeated, pkgName:%s", pkgName.c_str());
65             return ERR_DM_DISCOVERY_REPEATED;
66         }
67 
68         LOGI("DmDiscoveryManager::StartDeviceDiscovery stop preview discovery first, the preview pkgName is %s",
69             discoveryQueue_.front().c_str());
70         subscribeId = discoveryContextMap_[frontPkgName].subscribeId;
71     }
72     StopDeviceDiscovery(frontPkgName, subscribeId);
73     return DM_OK;
74 }
75 
StartDeviceDiscovery(const std::string & pkgName,const DmSubscribeInfo & subscribeInfo,const std::string & extra)76 int32_t DmDiscoveryManager::StartDeviceDiscovery(const std::string &pkgName, const DmSubscribeInfo &subscribeInfo,
77     const std::string &extra)
78 {
79     DmDeviceFilterOption dmFilter;
80     if (dmFilter.TransformToFilter(extra) != DM_OK) {
81         return ERR_DM_INPUT_PARA_INVALID;
82     }
83 
84     if (CheckDiscoveryQueue(pkgName) != DM_OK) {
85         return ERR_DM_DISCOVERY_REPEATED;
86     }
87     {
88         std::lock_guard<std::mutex> autoLock(locks_);
89         discoveryQueue_.push(pkgName);
90         DmDiscoveryContext context = {pkgName, extra, subscribeInfo.subscribeId, dmFilter.filterOp_, dmFilter.filters_};
91         discoveryContextMap_.emplace(pkgName, context);
92     }
93     softbusConnector_->RegisterSoftbusDiscoveryCallback(pkgName,
94         std::shared_ptr<ISoftbusDiscoveryCallback>(shared_from_this()));
95     CfgDiscoveryTimer();
96     return softbusConnector_->StartDiscovery(subscribeInfo);
97 }
98 
StartDeviceDiscovery(const std::string & pkgName,const uint16_t subscribeId,const std::string & filterOptions)99 int32_t DmDiscoveryManager::StartDeviceDiscovery(const std::string &pkgName, const uint16_t subscribeId,
100     const std::string &filterOptions)
101 {
102     DmDeviceFilterOption dmFilter;
103     dmFilter.TransformFilterOption(filterOptions);
104     if (CheckDiscoveryQueue(pkgName) != DM_OK) {
105         return ERR_DM_DISCOVERY_REPEATED;
106     }
107     {
108         std::lock_guard<std::mutex> autoLock(locks_);
109         discoveryQueue_.push(pkgName);
110         DmDiscoveryContext context = {pkgName, filterOptions, subscribeId, dmFilter.filterOp_, dmFilter.filters_};
111         discoveryContextMap_.emplace(pkgName, context);
112     }
113     softbusConnector_->RegisterSoftbusDiscoveryCallback(pkgName,
114         std::shared_ptr<ISoftbusDiscoveryCallback>(shared_from_this()));
115     CfgDiscoveryTimer();
116     return softbusConnector_->StartDiscovery(subscribeId);
117 }
118 
StopDeviceDiscovery(const std::string & pkgName,uint16_t subscribeId)119 int32_t DmDiscoveryManager::StopDeviceDiscovery(const std::string &pkgName, uint16_t subscribeId)
120 {
121     if (pkgName.empty()) {
122         LOGE("Invalid parameter, pkgName is empty.");
123         return ERR_DM_INPUT_PARA_INVALID;
124     }
125     {
126         std::lock_guard<std::mutex> autoLock(locks_);
127         if (!discoveryQueue_.empty()) {
128             discoveryQueue_.pop();
129         }
130         if (!discoveryContextMap_.empty()) {
131             discoveryContextMap_.erase(pkgName);
132             softbusConnector_->UnRegisterSoftbusDiscoveryCallback(pkgName);
133             timer_->DeleteTimer(std::string(DISCOVERY_TIMEOUT_TASK));
134         }
135     }
136     return softbusConnector_->StopDiscovery(subscribeId);
137 }
138 
OnDeviceFound(const std::string & pkgName,DmDeviceInfo & info,bool isOnline)139 void DmDiscoveryManager::OnDeviceFound(const std::string &pkgName, DmDeviceInfo &info, bool isOnline)
140 {
141     LOGI("DmDiscoveryManager::OnDeviceFound deviceId = %s", GetAnonyString(info.deviceId).c_str());
142     DmDiscoveryContext discoveryContext;
143     {
144         std::lock_guard<std::mutex> autoLock(locks_);
145         auto iter = discoveryContextMap_.find(pkgName);
146         if (iter == discoveryContextMap_.end()) {
147             LOGE("subscribeId not found by pkgName %s", GetAnonyString(pkgName).c_str());
148             return;
149         }
150         discoveryContext = iter->second;
151     }
152     DmDiscoveryFilter filter;
153     DmDeviceFilterPara filterPara;
154     filterPara.isOnline = isOnline;
155     filterPara.range = info.range;
156     filterPara.deviceType = info.deviceTypeId;
157     char localDeviceId[DEVICE_UUID_LENGTH];
158     GetDevUdid(localDeviceId, DEVICE_UUID_LENGTH);
159     info.authForm = DmAuthForm::INVALID_TYPE;
160     GetAuthForm(localDeviceId, info.deviceId, filterPara.isTrusted, info.authForm);
161     filterPara.authForm = info.authForm;
162     if (filter.IsValidDevice(discoveryContext.filterOp, discoveryContext.filters, filterPara)) {
163         listener_->OnDeviceFound(pkgName, discoveryContext.subscribeId, info);
164     }
165     return;
166 }
167 
GetAuthForm(const std::string & localDeviceId,const std::string & deviceId,bool & isTrusted,DmAuthForm & authForm)168 int32_t DmDiscoveryManager::GetAuthForm(const std::string &localDeviceId, const std::string &deviceId,
169     bool &isTrusted, DmAuthForm &authForm)
170 {
171     LOGI("Get localDeviceId: %s anth form.", GetAnonyString(localDeviceId).c_str());
172     isTrusted = false;
173     if (localDeviceId.empty() || deviceId.empty()) {
174         LOGE("Invalid parameter.");
175         return ERR_DM_INPUT_PARA_INVALID;
176     }
177     if (hiChainConnector_ == nullptr || softbusConnector_ == nullptr) {
178         LOGE("hiChainConnector_ or softbusConnector_ is nullpter.");
179         return ERR_DM_POINT_NULL;
180     }
181 
182     std::vector<std::string> trustDeviceUdidList = hiChainConnector_->GetTrustedDevices(localDeviceId);
183     if (trustDeviceUdidList.empty()) {
184         LOGI("Trusted devices is empty.");
185         return DM_OK;
186     }
187     std::string udidHash;
188     for (auto udid : trustDeviceUdidList) {
189         udidHash = softbusConnector_->GetDeviceUdidHashByUdid(udid);
190         if (udidHash == deviceId) {
191             isTrusted = true;
192             authForm = hiChainConnector_->GetGroupType(udid);
193             LOGI("deviceId: %s is trusted!", GetAnonyString(deviceId).c_str());
194         }
195     }
196 
197     return DM_OK;
198 }
199 
OnDeviceFound(const std::string & pkgName,DmDeviceBasicInfo & info,const int32_t range,bool isOnline)200 void DmDiscoveryManager::OnDeviceFound(const std::string &pkgName,
201     DmDeviceBasicInfo &info, const int32_t range, bool isOnline)
202 {
203     LOGI("DmDiscoveryManager::OnDeviceFound deviceId = %s,isOnline %d",
204         GetAnonyString(info.deviceId).c_str(), isOnline);
205     DmDiscoveryContext discoveryContext;
206     {
207         std::lock_guard<std::mutex> autoLock(locks_);
208         auto iter = discoveryContextMap_.find(pkgName);
209         if (iter == discoveryContextMap_.end()) {
210             LOGE("subscribeId not found by pkgName %s", GetAnonyString(pkgName).c_str());
211             return;
212         }
213         discoveryContext = iter->second;
214     }
215     DmDiscoveryFilter filter;
216     DmDeviceFilterPara filterPara;
217     filterPara.isOnline = isOnline;
218     filterPara.range = range;
219     filterPara.deviceType = info.deviceTypeId;
220     char localDeviceId[DEVICE_UUID_LENGTH];
221     GetDevUdid(localDeviceId, DEVICE_UUID_LENGTH);
222     DmAuthForm authForm = DmAuthForm::INVALID_TYPE;
223     GetAuthForm(localDeviceId, info.deviceId, filterPara.isTrusted, authForm);
224     filterPara.authForm = authForm;
225     if (filter.IsValidDevice(discoveryContext.filterOp, discoveryContext.filters, filterPara)) {
226         listener_->OnDeviceFound(pkgName, discoveryContext.subscribeId, info);
227     }
228     return;
229 }
230 
OnDiscoveryFailed(const std::string & pkgName,int32_t subscribeId,int32_t failedReason)231 void DmDiscoveryManager::OnDiscoveryFailed(const std::string &pkgName, int32_t subscribeId, int32_t failedReason)
232 {
233     LOGI("DmDiscoveryManager::OnDiscoveryFailed subscribeId = %d reason = %d", subscribeId, failedReason);
234     StopDeviceDiscovery(pkgName, (uint32_t)subscribeId);
235     listener_->OnDiscoveryFailed(pkgName, (uint32_t)subscribeId, failedReason);
236 }
237 
OnDiscoverySuccess(const std::string & pkgName,int32_t subscribeId)238 void DmDiscoveryManager::OnDiscoverySuccess(const std::string &pkgName, int32_t subscribeId)
239 {
240     LOGI("DmDiscoveryManager::OnDiscoverySuccess subscribeId = %d", subscribeId);
241     {
242         std::lock_guard<std::mutex> autoLock(locks_);
243         discoveryContextMap_[pkgName].subscribeId = (uint32_t)subscribeId;
244     }
245     listener_->OnDiscoverySuccess(pkgName, subscribeId);
246 }
247 
HandleDiscoveryTimeout(std::string name)248 void DmDiscoveryManager::HandleDiscoveryTimeout(std::string name)
249 {
250     (void)name;
251     LOGI("DmDiscoveryManager::HandleDiscoveryTimeout");
252     uint16_t subscribeId = 0;
253     std::string pkgName = "";
254     {
255         std::lock_guard<std::mutex> autoLock(locks_);
256         if (discoveryQueue_.empty()) {
257             LOGE("HandleDiscoveryTimeout: discovery queue is empty.");
258             return;
259         }
260 
261         pkgName = discoveryQueue_.front();
262         auto iter = discoveryContextMap_.find(pkgName);
263         if (iter == discoveryContextMap_.end()) {
264             LOGE("HandleDiscoveryTimeout: subscribeId not found by pkgName %s", GetAnonyString(pkgName).c_str());
265             return;
266         }
267         subscribeId = discoveryContextMap_[pkgName].subscribeId;
268     }
269     StopDeviceDiscovery(pkgName, subscribeId);
270 }
271 } // namespace DistributedHardware
272 } // namespace OHOS
273