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 #define LOG_TAG "processCommunication"
17
18 #include "communicator_context.h"
19 #include "device_manager_adapter.h"
20 #include "log_print.h"
21 #include "softbus_adapter.h"
22 #include "process_communicator_impl.h"
23 namespace OHOS {
24 namespace AppDistributedKv {
25 using namespace DistributedDB;
26 using namespace OHOS::DistributedData;
27 using DmAdapter = OHOS::DistributedData::DeviceManagerAdapter;
ProcessCommunicatorImpl()28 ProcessCommunicatorImpl::ProcessCommunicatorImpl()
29 {
30 }
31
ProcessCommunicatorImpl(RouteHeadHandlerCreator handlerCreator)32 ProcessCommunicatorImpl::ProcessCommunicatorImpl(RouteHeadHandlerCreator handlerCreator)
33 : routeHeadHandlerCreator_(std::move(handlerCreator))
34 {
35 }
36
~ProcessCommunicatorImpl()37 ProcessCommunicatorImpl::~ProcessCommunicatorImpl()
38 {
39 ZLOGE("destructor.");
40 }
41
Start(const std::string & processLabel)42 DBStatus ProcessCommunicatorImpl::Start(const std::string &processLabel)
43 {
44 ZLOGI("init commProvider");
45 thisProcessLabel_ = processLabel;
46 PipeInfo pi = {thisProcessLabel_, ""};
47 Status errCode = CommunicationProvider::GetInstance().Start(pi);
48 if (errCode != Status::SUCCESS) {
49 ZLOGE("commProvider_ Start Fail.");
50 return DBStatus::DB_ERROR;
51 }
52 return DBStatus::OK;
53 }
54
Stop()55 DBStatus ProcessCommunicatorImpl::Stop()
56 {
57 PipeInfo pi = {thisProcessLabel_, ""};
58 Status errCode = CommunicationProvider::GetInstance().Stop(pi);
59 if (errCode != Status::SUCCESS) {
60 ZLOGE("commProvider_ Stop Fail.");
61 return DBStatus::DB_ERROR;
62 }
63 return DBStatus::OK;
64 }
65
RegOnDeviceChange(const OnDeviceChange & callback)66 DBStatus ProcessCommunicatorImpl::RegOnDeviceChange(const OnDeviceChange &callback)
67 {
68 {
69 std::lock_guard<std::mutex> onDeviceChangeLockGard(onDeviceChangeMutex_);
70 onDeviceChangeHandler_ = callback;
71 }
72
73 PipeInfo pi = {thisProcessLabel_, ""};
74 if (callback) {
75 Status errCode = DmAdapter::GetInstance().StartWatchDeviceChange(this, pi);
76 if (errCode != Status::SUCCESS) {
77 ZLOGE("commProvider_ StartWatchDeviceChange Fail.");
78 return DBStatus::DB_ERROR;
79 }
80 } else {
81 Status errCode = DmAdapter::GetInstance().StopWatchDeviceChange(this, pi);
82 if (errCode != Status::SUCCESS) {
83 ZLOGE("commProvider_ StopWatchDeviceChange Fail.");
84 return DBStatus::DB_ERROR;
85 }
86 }
87
88 return DBStatus::OK;
89 }
90
RegOnDataReceive(const OnDataReceive & callback)91 DBStatus ProcessCommunicatorImpl::RegOnDataReceive(const OnDataReceive &callback)
92 {
93 {
94 std::lock_guard<std::mutex> onDataReceiveLockGard(onDataReceiveMutex_);
95 onDataReceiveHandler_ = callback;
96 }
97
98 PipeInfo pi = {thisProcessLabel_, ""};
99 if (callback) {
100 Status errCode = CommunicationProvider::GetInstance().StartWatchDataChange(this, pi);
101 if (errCode != Status::SUCCESS) {
102 ZLOGE("commProvider_ StartWatchDataChange Fail.");
103 return DBStatus::DB_ERROR;
104 }
105 } else {
106 Status errCode = CommunicationProvider::GetInstance().StopWatchDataChange(this, pi);
107 if (errCode != Status::SUCCESS) {
108 ZLOGE("commProvider_ StopWatchDataChange Fail.");
109 return DBStatus::DB_ERROR;
110 }
111 }
112 return DBStatus::OK;
113 }
114
RegOnSendAble(const OnSendAble & sendAbleCallback)115 void ProcessCommunicatorImpl::RegOnSendAble(const OnSendAble &sendAbleCallback)
116 {
117 {
118 std::lock_guard<decltype(sessionMutex_)> lock(sessionMutex_);
119 sessionListener_ = sendAbleCallback;
120 }
121 if (!sendAbleCallback) {
122 ZLOGE("send callback is nullptr.");
123 return;
124 }
125 auto status = CommunicatorContext::GetInstance().RegSessionListener(this);
126 ZLOGD("reg status:%{public}d", status);
127 }
128
SendData(const DeviceInfos & dstDevInfo,const uint8_t * data,uint32_t length)129 DBStatus ProcessCommunicatorImpl::SendData(const DeviceInfos &dstDevInfo, const uint8_t *data, uint32_t length)
130 {
131 uint32_t totalLength = 0;
132 return SendData(dstDevInfo, data, length, totalLength);
133 }
134
SendData(const DeviceInfos & dstDevInfo,const uint8_t * data,uint32_t length,uint32_t totalLength)135 DBStatus ProcessCommunicatorImpl::SendData(const DeviceInfos &dstDevInfo, const uint8_t *data, uint32_t length,
136 uint32_t totalLength)
137 {
138 PipeInfo pi = {thisProcessLabel_, ""};
139 const DataInfo dataInfo = { const_cast<uint8_t *>(data), length};
140 DeviceId destination;
141 destination.deviceId = dstDevInfo.identifier;
142 auto [errCode, softBusErrCode] =
143 CommunicationProvider::GetInstance().SendData(pi, destination, dataInfo, totalLength);
144 if (errCode == Status::RATE_LIMIT) {
145 ZLOGD("commProvider_ opening session, status:%{public}d.", static_cast<int>(softBusErrCode));
146 return DBStatus::RATE_LIMIT;
147 }
148 if (errCode != Status::SUCCESS) {
149 ZLOGE("commProvider_ SendData Fail. code:%{public}d", softBusErrCode);
150 if (softBusErrCode == 0) {
151 return DBStatus::DB_ERROR;
152 }
153 return static_cast<DBStatus>(softBusErrCode);
154 }
155 return DBStatus::OK;
156 }
157
GetMtuSize()158 uint32_t ProcessCommunicatorImpl::GetMtuSize()
159 {
160 return MTU_SIZE;
161 }
162
GetMtuSize(const DeviceInfos & devInfo)163 uint32_t ProcessCommunicatorImpl::GetMtuSize(const DeviceInfos &devInfo)
164 {
165 return SoftBusAdapter::GetInstance()->GetMtuSize({ devInfo.identifier });
166 }
167
GetTimeout(const DeviceInfos & devInfo)168 uint32_t ProcessCommunicatorImpl::GetTimeout(const DeviceInfos &devInfo)
169 {
170 return SoftBusAdapter::GetInstance()->GetTimeout({ devInfo.identifier });
171 }
172
GetLocalDeviceInfos()173 DeviceInfos ProcessCommunicatorImpl::GetLocalDeviceInfos()
174 {
175 DeviceInfos localDevInfos;
176 DeviceInfo devInfo = DmAdapter::GetInstance().GetLocalDevice();
177 localDevInfos.identifier = devInfo.uuid;
178 return localDevInfos;
179 }
180
GetRemoteOnlineDeviceInfosList()181 std::vector<DeviceInfos> ProcessCommunicatorImpl::GetRemoteOnlineDeviceInfosList()
182 {
183 std::vector<DeviceInfos> remoteDevInfos;
184 std::vector<DeviceInfo> devInfoVec = DmAdapter::GetInstance().GetRemoteDevices();
185 for (auto const &entry : devInfoVec) {
186 DeviceInfos remoteDev;
187 remoteDev.identifier = entry.uuid;
188 remoteDevInfos.push_back(remoteDev);
189 }
190 return remoteDevInfos;
191 }
192
IsSameProcessLabelStartedOnPeerDevice(const DeviceInfos & peerDevInfo)193 bool ProcessCommunicatorImpl::IsSameProcessLabelStartedOnPeerDevice(const DeviceInfos &peerDevInfo)
194 {
195 PipeInfo pi = {thisProcessLabel_, ""};
196 DeviceId di = {peerDevInfo.identifier};
197 return CommunicationProvider::GetInstance().IsSameStartedOnPeer(pi, di);
198 }
199
OnMessage(const DeviceInfo & info,const uint8_t * ptr,const int size,const PipeInfo & pipeInfo) const200 void ProcessCommunicatorImpl::OnMessage(const DeviceInfo &info, const uint8_t *ptr, const int size,
201 __attribute__((unused)) const PipeInfo &pipeInfo) const
202 {
203 std::lock_guard<std::mutex> onDataReceiveLockGuard(onDataReceiveMutex_);
204 if (onDataReceiveHandler_ == nullptr) {
205 ZLOGE("onDataReceiveHandler_ invalid.");
206 return;
207 }
208 DeviceInfos devInfo;
209 devInfo.identifier = info.uuid;
210 onDataReceiveHandler_(devInfo, ptr, static_cast<uint32_t>(size));
211 }
212
OnDeviceChanged(const DeviceInfo & info,const DeviceChangeType & type) const213 void ProcessCommunicatorImpl::OnDeviceChanged(const DeviceInfo &info, const DeviceChangeType &type) const
214 {
215 if (type == DeviceChangeType::DEVICE_ONREADY || info.uuid == DmAdapter::CLOUD_DEVICE_UUID) {
216 return;
217 }
218 std::lock_guard<std::mutex> onDeviceChangeLockGuard(onDeviceChangeMutex_);
219 if (onDeviceChangeHandler_ == nullptr) {
220 ZLOGE("onDeviceChangeHandler_ invalid.");
221 return;
222 }
223 DeviceInfos devInfo;
224 devInfo.identifier = info.uuid;
225 onDeviceChangeHandler_(devInfo, (type == DeviceChangeType::DEVICE_ONLINE));
226 }
227
OnSessionReady(const DeviceInfo & info,int32_t errCode) const228 void ProcessCommunicatorImpl::OnSessionReady(const DeviceInfo &info, int32_t errCode) const
229 {
230 std::lock_guard<decltype(sessionMutex_)> lock(sessionMutex_);
231 if (sessionListener_ == nullptr) {
232 return;
233 }
234 DeviceInfos devInfos;
235 devInfos.identifier = info.uuid;
236 sessionListener_(devInfos, errCode);
237 }
238
GetExtendHeaderHandle(const ExtendInfo & info)239 std::shared_ptr<ExtendHeaderHandle> ProcessCommunicatorImpl::GetExtendHeaderHandle(const ExtendInfo &info)
240 {
241 if (routeHeadHandlerCreator_ != nullptr) {
242 return routeHeadHandlerCreator_(info);
243 }
244 return {};
245 }
246
CheckAndGetDataHeadInfo(const uint8_t * data,uint32_t dataLen,uint32_t & headLen,std::vector<std::string> & users)247 DBStatus ProcessCommunicatorImpl::CheckAndGetDataHeadInfo(
248 const uint8_t *data, uint32_t dataLen, uint32_t &headLen, std::vector<std::string> &users)
249 {
250 ZLOGD("begin");
251 if (routeHeadHandlerCreator_ == nullptr) {
252 ZLOGE("header handler creator not registered");
253 return DBStatus::DB_ERROR;
254 }
255 auto handler = routeHeadHandlerCreator_({});
256 if (handler == nullptr) {
257 ZLOGE("failed to get header handler");
258 return DBStatus::DB_ERROR;
259 }
260 auto ret = handler->ParseHeadData(data, dataLen, headLen, users);
261 if (!ret) {
262 ZLOGD("illegal head format");
263 return DBStatus::INVALID_FORMAT;
264 }
265 if (users.empty()) {
266 ZLOGW("no valid user");
267 return DBStatus::NO_PERMISSION;
268 }
269 ZLOGD("ok, result:%{public}zu, user:%{public}s", users.size(), users.front().c_str());
270 return DBStatus::OK;
271 }
272 } // namespace AppDistributedKv
273 } // namespace OHOS