1c29fa5a6Sopenharmony_ci/*
2c29fa5a6Sopenharmony_ci * Copyright (c) 2023 Huawei Device Co., Ltd.
3c29fa5a6Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4c29fa5a6Sopenharmony_ci * you may not use this file except in compliance with the License.
5c29fa5a6Sopenharmony_ci * You may obtain a copy of the License at
6c29fa5a6Sopenharmony_ci *
7c29fa5a6Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8c29fa5a6Sopenharmony_ci *
9c29fa5a6Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10c29fa5a6Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11c29fa5a6Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12c29fa5a6Sopenharmony_ci * See the License for the specific language governing permissions and
13c29fa5a6Sopenharmony_ci * limitations under the License.
14c29fa5a6Sopenharmony_ci */
15c29fa5a6Sopenharmony_ci
16c29fa5a6Sopenharmony_ci#include "socket_connection.h"
17c29fa5a6Sopenharmony_ci
18c29fa5a6Sopenharmony_ci#include <sys/socket.h>
19c29fa5a6Sopenharmony_ci#include <unistd.h>
20c29fa5a6Sopenharmony_ci
21c29fa5a6Sopenharmony_ci#include "devicestatus_define.h"
22c29fa5a6Sopenharmony_ci#include "include/util.h"
23c29fa5a6Sopenharmony_ci
24c29fa5a6Sopenharmony_ci#undef LOG_TAG
25c29fa5a6Sopenharmony_ci#define LOG_TAG "SocketConnection"
26c29fa5a6Sopenharmony_ci
27c29fa5a6Sopenharmony_cinamespace OHOS {
28c29fa5a6Sopenharmony_cinamespace Msdp {
29c29fa5a6Sopenharmony_cinamespace DeviceStatus {
30c29fa5a6Sopenharmony_ci
31c29fa5a6Sopenharmony_ciSocketConnection::SocketConnection(int32_t socketFd,
32c29fa5a6Sopenharmony_ci                                   std::function<void(NetPacket&)> recv,
33c29fa5a6Sopenharmony_ci                                   std::function<void()> onDisconnected)
34c29fa5a6Sopenharmony_ci    : socketFd_(socketFd), recv_(recv), onDisconnected_(onDisconnected)
35c29fa5a6Sopenharmony_ci{}
36c29fa5a6Sopenharmony_ci
37c29fa5a6Sopenharmony_ciSocketConnection::~SocketConnection()
38c29fa5a6Sopenharmony_ci{
39c29fa5a6Sopenharmony_ci    if ((socketFd_ >= 0) && (::close(socketFd_) != 0)) {
40c29fa5a6Sopenharmony_ci        FI_HILOGE("close(%{public}d) failed:%{public}s", socketFd_, ::strerror(errno));
41c29fa5a6Sopenharmony_ci    }
42c29fa5a6Sopenharmony_ci}
43c29fa5a6Sopenharmony_ci
44c29fa5a6Sopenharmony_cistd::shared_ptr<SocketConnection> SocketConnection::Connect(std::function<int32_t()> socket,
45c29fa5a6Sopenharmony_ci    std::function<void(NetPacket&)> recv, std::function<void()> onDisconnected)
46c29fa5a6Sopenharmony_ci{
47c29fa5a6Sopenharmony_ci    CALL_DEBUG_ENTER;
48c29fa5a6Sopenharmony_ci    CHKPP(socket);
49c29fa5a6Sopenharmony_ci    int32_t sockFd = socket();
50c29fa5a6Sopenharmony_ci    if (sockFd < 0) {
51c29fa5a6Sopenharmony_ci        return nullptr;
52c29fa5a6Sopenharmony_ci    }
53c29fa5a6Sopenharmony_ci    return std::make_shared<SocketConnection>(sockFd, recv, onDisconnected);
54c29fa5a6Sopenharmony_ci}
55c29fa5a6Sopenharmony_ci
56c29fa5a6Sopenharmony_civoid SocketConnection::OnReadable(int32_t fd)
57c29fa5a6Sopenharmony_ci{
58c29fa5a6Sopenharmony_ci    CALL_DEBUG_ENTER;
59c29fa5a6Sopenharmony_ci    char buf[MAX_PACKET_BUF_SIZE] {};
60c29fa5a6Sopenharmony_ci    ssize_t numRead;
61c29fa5a6Sopenharmony_ci
62c29fa5a6Sopenharmony_ci    do {
63c29fa5a6Sopenharmony_ci        numRead = ::recv(fd, buf, sizeof(buf), MSG_DONTWAIT);
64c29fa5a6Sopenharmony_ci        if (numRead > 0) {
65c29fa5a6Sopenharmony_ci            buffer_.Write(buf, numRead);
66c29fa5a6Sopenharmony_ci            OnReadPackets(buffer_, recv_);
67c29fa5a6Sopenharmony_ci        } else if (numRead < 0) {
68c29fa5a6Sopenharmony_ci            if (errno == EINTR) {
69c29fa5a6Sopenharmony_ci                FI_HILOGD("recv was interrupted, read again");
70c29fa5a6Sopenharmony_ci                continue;
71c29fa5a6Sopenharmony_ci            }
72c29fa5a6Sopenharmony_ci            if ((errno == EAGAIN) || (errno == EWOULDBLOCK)) {
73c29fa5a6Sopenharmony_ci                FI_HILOGW("No available data");
74c29fa5a6Sopenharmony_ci            } else {
75c29fa5a6Sopenharmony_ci                FI_HILOGE("recv failed:%{public}s", ::strerror(errno));
76c29fa5a6Sopenharmony_ci            }
77c29fa5a6Sopenharmony_ci            break;
78c29fa5a6Sopenharmony_ci        } else {
79c29fa5a6Sopenharmony_ci            FI_HILOGE("EOF happened");
80c29fa5a6Sopenharmony_ci            OnShutdown(fd);
81c29fa5a6Sopenharmony_ci            break;
82c29fa5a6Sopenharmony_ci        }
83c29fa5a6Sopenharmony_ci    } while (numRead == sizeof(buf));
84c29fa5a6Sopenharmony_ci}
85c29fa5a6Sopenharmony_ci
86c29fa5a6Sopenharmony_civoid SocketConnection::OnShutdown(int32_t fd)
87c29fa5a6Sopenharmony_ci{
88c29fa5a6Sopenharmony_ci    if (onDisconnected_) {
89c29fa5a6Sopenharmony_ci        onDisconnected_();
90c29fa5a6Sopenharmony_ci    }
91c29fa5a6Sopenharmony_ci}
92c29fa5a6Sopenharmony_ci
93c29fa5a6Sopenharmony_civoid SocketConnection::OnException(int32_t fd)
94c29fa5a6Sopenharmony_ci{
95c29fa5a6Sopenharmony_ci    OnShutdown(fd);
96c29fa5a6Sopenharmony_ci}
97c29fa5a6Sopenharmony_ci} // namespace DeviceStatus
98c29fa5a6Sopenharmony_ci} // namespace Msdp
99c29fa5a6Sopenharmony_ci} // namespace OHOS