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 #include "sensor_basic_data_channel.h"
17
18 #include <fcntl.h>
19 #include <sys/socket.h>
20 #include <unistd.h>
21
22 #include "dmd_report.h"
23 #include "sensors_errors.h"
24 #include "sensors_log_domain.h"
25
26 namespace OHOS {
27 namespace Sensors {
28 using namespace OHOS::HiviewDFX;
29
30 namespace {
31 constexpr HiLogLabel LABEL = { LOG_CORE, SensorsLogDomain::SENSOR_UTILS, "SensorBasicChannel" };
32 constexpr int32_t DEFAULT_CHANNEL_SIZE = 2 * 1024;
33 constexpr int32_t SOCKET_PAIR_SIZE = 2;
34 } // namespace
35
SensorBasicDataChannel()36 SensorBasicDataChannel::SensorBasicDataChannel() : sendFd_(INVALID_FD), receiveFd_(INVALID_FD), isActive_(false)
37 {
38 HiLog::Debug(LABEL, "%{public}s isActive_ : %{public}d, sendFd: %{public}d", __func__, isActive_, sendFd_);
39 }
40
CreateSensorBasicChannel(size_t sendSize,size_t receiveSize)41 int32_t SensorBasicDataChannel::CreateSensorBasicChannel(size_t sendSize, size_t receiveSize)
42 {
43 if ((sendFd_ != INVALID_FD) || (receiveFd_ != INVALID_FD)) {
44 HiLog::Debug(LABEL, "%{public}s already create socketpair", __func__);
45 return ERR_OK;
46 }
47
48 int32_t socketPair[SOCKET_PAIR_SIZE] = { 0 };
49 if (socketpair(AF_UNIX, SOCK_SEQPACKET, 0, socketPair) != 0) {
50 DmdReport::ReportException(SENSOR_DATA_CHANNEL_EXCEPTION, "CreateSensorBasicChannel",
51 SENSOR_CHANNEL_SOCKET_CREATE_ERR);
52 HiLog::Error(LABEL, "%{public}s create socketpair failed", __func__);
53 sendFd_ = INVALID_FD;
54 receiveFd_ = INVALID_FD;
55 return SENSOR_CHANNEL_SOCKET_CREATE_ERR;
56 }
57 // set socket attr
58 setsockopt(socketPair[0], SOL_SOCKET, SO_SNDBUF, &sendSize, sizeof(sendSize));
59 setsockopt(socketPair[1], SOL_SOCKET, SO_RCVBUF, &receiveSize, sizeof(receiveSize));
60 int32_t bufferSize = DEFAULT_CHANNEL_SIZE;
61 int32_t ret = setsockopt(socketPair[0], SOL_SOCKET, SO_RCVBUF, &bufferSize, sizeof(bufferSize));
62 if (ret != 0) {
63 HiLog::Error(LABEL, "%{public}s setsockopt socketpair 0 failed", __func__);
64 return SENSOR_CHANNEL_SOCKET_CREATE_ERR;
65 }
66 ret = setsockopt(socketPair[1], SOL_SOCKET, SO_SNDBUF, &bufferSize, sizeof(bufferSize));
67 if (ret != 0) {
68 HiLog::Error(LABEL, "%{public}s setsockopt socketpair 1 failed", __func__);
69 return SENSOR_CHANNEL_SOCKET_CREATE_ERR;
70 }
71 ret = fcntl(socketPair[0], F_SETFL, O_NONBLOCK);
72 if (ret != 0) {
73 HiLog::Error(LABEL, "%{public}s fcntl socketpair 0 failed", __func__);
74 return SENSOR_CHANNEL_SOCKET_CREATE_ERR;
75 }
76 ret = fcntl(socketPair[1], F_SETFL, O_NONBLOCK);
77 if (ret != 0) {
78 HiLog::Error(LABEL, "%{public}s fcntl socketpair 1 failed", __func__);
79 return SENSOR_CHANNEL_SOCKET_CREATE_ERR;
80 }
81 sendFd_ = socketPair[0];
82 receiveFd_ = socketPair[1];
83 HiLog::Debug(LABEL, "%{public}s create socketpair success, receiveFd_ : %{public}d, sendFd_ : %{public}d", __func__,
84 receiveFd_, sendFd_);
85 return ERR_OK;
86 }
87
CreateSensorBasicChannel(MessageParcel & data)88 int32_t SensorBasicDataChannel::CreateSensorBasicChannel(MessageParcel &data)
89 {
90 HiLog::Debug(LABEL, "%{public}s begin", __func__);
91 if ((sendFd_ != INVALID_FD) || (receiveFd_ != INVALID_FD)) {
92 HiLog::Debug(LABEL, "%{public}s already create socketpair", __func__);
93 return ERR_OK;
94 }
95 int32_t tmpFd = data.ReadFileDescriptor();
96 if (tmpFd < 0) {
97 HiLog::Error(LABEL, "%{public}s ReadFileDescriptor failed", __func__);
98 sendFd_ = INVALID_FD;
99 return SENSOR_CHANNEL_DUP_ERR;
100 }
101 sendFd_ = dup(tmpFd);
102 if (sendFd_ < 0) {
103 HiLog::Error(LABEL, "%{public}s dup FileDescriptor failed", __func__);
104 sendFd_ = INVALID_FD;
105 return SENSOR_CHANNEL_DUP_ERR;
106 }
107 return ERR_OK;
108 }
109
~SensorBasicDataChannel()110 SensorBasicDataChannel::~SensorBasicDataChannel()
111 {
112 DestroySensorBasicChannel();
113 }
114
SendToBinder(MessageParcel & data)115 int32_t SensorBasicDataChannel::SendToBinder(MessageParcel &data)
116 {
117 HiLog::Debug(LABEL, "%{public}s sendFd: %{public}d", __func__, sendFd_);
118 if (sendFd_ < 0) {
119 HiLog::Error(LABEL, "%{public}s sendFd FileDescriptor error", __func__);
120 return SENSOR_CHANNEL_SENDFD_ERR;
121 }
122 bool result = data.WriteFileDescriptor(sendFd_);
123 if (!result) {
124 HiLog::Error(LABEL, "%{public}s send sendFd_ failed", __func__);
125 CloseSendFd();
126 return SENSOR_CHANNEL_WRITE_DESCRIPTOR_ERR;
127 }
128 return ERR_OK;
129 }
130
CloseSendFd()131 void SensorBasicDataChannel::CloseSendFd()
132 {
133 if (sendFd_ != INVALID_FD) {
134 close(sendFd_);
135 sendFd_ = INVALID_FD;
136 HiLog::Debug(LABEL, "%{public}s close sendFd_", __func__);
137 }
138 }
139
SendData(const void * vaddr,size_t size)140 int32_t SensorBasicDataChannel::SendData(const void *vaddr, size_t size)
141 {
142 if (vaddr == nullptr || sendFd_ < 0) {
143 HiLog::Error(LABEL, "%{public}s failed, param is invalid", __func__);
144 return SENSOR_CHANNEL_SEND_ADDR_ERR;
145 }
146 ssize_t length;
147 do {
148 length = send(sendFd_, vaddr, size, MSG_DONTWAIT | MSG_NOSIGNAL);
149 } while (errno == EINTR);
150 if (length < 0) {
151 HiLog::Error(LABEL, "%{public}s send fail : %{public}d, length = %{public}d", __func__, errno, (int32_t)length);
152 return SENSOR_CHANNEL_SEND_DATA_ERR;
153 }
154 return ERR_OK;
155 }
156
ReceiveData(void * vaddr,size_t size)157 int32_t SensorBasicDataChannel::ReceiveData(void *vaddr, size_t size)
158 {
159 if (vaddr == nullptr || (receiveFd_ == INVALID_FD)) {
160 HiLog::Error(LABEL, "%{public}s failed, vaddr is null or receiveFd_ invalid", __func__);
161 return SENSOR_CHANNEL_RECEIVE_ADDR_ERR;
162 }
163 ssize_t length;
164 do {
165 length = recv(receiveFd_, vaddr, size, MSG_DONTWAIT);
166 } while (errno == EINTR);
167 if (length < 0) {
168 return 0;
169 }
170 return length;
171 }
172
GetSendDataFd() const173 int32_t SensorBasicDataChannel::GetSendDataFd() const
174 {
175 return sendFd_;
176 }
177
GetReceiveDataFd() const178 int32_t SensorBasicDataChannel::GetReceiveDataFd() const
179 {
180 return receiveFd_;
181 }
182
DestroySensorBasicChannel()183 int32_t SensorBasicDataChannel::DestroySensorBasicChannel()
184 {
185 if (sendFd_ >= 0) {
186 close(sendFd_);
187 sendFd_ = INVALID_FD;
188 HiLog::Debug(LABEL, "%{public}s close sendFd_ success", __func__);
189 }
190 if (receiveFd_ >= 0) {
191 close(receiveFd_);
192 receiveFd_ = INVALID_FD;
193 HiLog::Debug(LABEL, "%{public}s close receiveFd_ success", __func__);
194 }
195 return ERR_OK;
196 }
197
GetDataCacheBuf() const198 const std::unordered_map<uint32_t, struct SensorEvent> &SensorBasicDataChannel::GetDataCacheBuf() const
199 {
200 return dataCacheBuf_;
201 }
202
GetSensorStatus() const203 bool SensorBasicDataChannel::GetSensorStatus() const
204 {
205 return isActive_;
206 }
207
SetSensorStatus(bool isActive)208 void SensorBasicDataChannel::SetSensorStatus(bool isActive)
209 {
210 HiLog::Debug(LABEL, "%{public}s isActive_ : %{public}d", __func__, isActive);
211 std::unique_lock<std::mutex> lock(statusLock_);
212 isActive_ = isActive;
213 return;
214 }
215 } // namespace Sensors
216 } // namespace OHOS
217