• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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