1c29fa5a6Sopenharmony_ci/*
2c29fa5a6Sopenharmony_ci * Copyright (c) 2021-2022 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 "uds_socket.h"
17c29fa5a6Sopenharmony_ci
18c29fa5a6Sopenharmony_ci#include <cinttypes>
19c29fa5a6Sopenharmony_ci
20c29fa5a6Sopenharmony_ci#include "error_multimodal.h"
21c29fa5a6Sopenharmony_ci#include "mmi_log.h"
22c29fa5a6Sopenharmony_ci
23c29fa5a6Sopenharmony_ci#undef MMI_LOG_TAG
24c29fa5a6Sopenharmony_ci#define MMI_LOG_TAG "UDSSocket"
25c29fa5a6Sopenharmony_ci
26c29fa5a6Sopenharmony_cinamespace OHOS {
27c29fa5a6Sopenharmony_cinamespace MMI {
28c29fa5a6Sopenharmony_ciUDSSocket::UDSSocket() {}
29c29fa5a6Sopenharmony_ci
30c29fa5a6Sopenharmony_ciUDSSocket::~UDSSocket()
31c29fa5a6Sopenharmony_ci{
32c29fa5a6Sopenharmony_ci    Close();
33c29fa5a6Sopenharmony_ci    EpollClose();
34c29fa5a6Sopenharmony_ci}
35c29fa5a6Sopenharmony_ci
36c29fa5a6Sopenharmony_ciint32_t UDSSocket::EpollCreate(int32_t size)
37c29fa5a6Sopenharmony_ci{
38c29fa5a6Sopenharmony_ci    epollFd_ = epoll_create(size);
39c29fa5a6Sopenharmony_ci    if (epollFd_ < 0) {
40c29fa5a6Sopenharmony_ci        MMI_HILOGE("epoll_create return %{public}d", epollFd_);
41c29fa5a6Sopenharmony_ci    } else {
42c29fa5a6Sopenharmony_ci        MMI_HILOGI("epoll_create, epollFd_:%{public}d", epollFd_);
43c29fa5a6Sopenharmony_ci    }
44c29fa5a6Sopenharmony_ci    return epollFd_;
45c29fa5a6Sopenharmony_ci}
46c29fa5a6Sopenharmony_ci
47c29fa5a6Sopenharmony_ciint32_t UDSSocket::EpollCtl(int32_t fd, int32_t op, struct epoll_event &event, int32_t epollFd)
48c29fa5a6Sopenharmony_ci{
49c29fa5a6Sopenharmony_ci    if (fd < 0) {
50c29fa5a6Sopenharmony_ci        MMI_HILOGE("Invalid fd");
51c29fa5a6Sopenharmony_ci        return RET_ERR;
52c29fa5a6Sopenharmony_ci    }
53c29fa5a6Sopenharmony_ci    if (epollFd < 0) {
54c29fa5a6Sopenharmony_ci        epollFd = epollFd_;
55c29fa5a6Sopenharmony_ci    }
56c29fa5a6Sopenharmony_ci    if (epollFd < 0) {
57c29fa5a6Sopenharmony_ci        MMI_HILOGE("Invalid param epollFd");
58c29fa5a6Sopenharmony_ci        return RET_ERR;
59c29fa5a6Sopenharmony_ci    }
60c29fa5a6Sopenharmony_ci    int32_t ret;
61c29fa5a6Sopenharmony_ci    if (op == EPOLL_CTL_DEL) {
62c29fa5a6Sopenharmony_ci        ret = epoll_ctl(epollFd, op, fd, nullptr);
63c29fa5a6Sopenharmony_ci    } else {
64c29fa5a6Sopenharmony_ci        ret = epoll_ctl(epollFd, op, fd, &event);
65c29fa5a6Sopenharmony_ci    }
66c29fa5a6Sopenharmony_ci    if (ret < 0) {
67c29fa5a6Sopenharmony_ci        MMI_HILOGE("epoll_ctl return %{public}d,epollFd_:%{public}d,"
68c29fa5a6Sopenharmony_ci                   "op:%{public}d,fd:%{public}d,errno:%{public}d",
69c29fa5a6Sopenharmony_ci                   ret, epollFd, op, fd, errno);
70c29fa5a6Sopenharmony_ci    }
71c29fa5a6Sopenharmony_ci    return ret;
72c29fa5a6Sopenharmony_ci}
73c29fa5a6Sopenharmony_ci
74c29fa5a6Sopenharmony_ciint32_t UDSSocket::EpollWait(struct epoll_event &events, int32_t maxevents, int32_t timeout, int32_t epollFd)
75c29fa5a6Sopenharmony_ci{
76c29fa5a6Sopenharmony_ci    if (epollFd < 0) {
77c29fa5a6Sopenharmony_ci        epollFd = epollFd_;
78c29fa5a6Sopenharmony_ci    }
79c29fa5a6Sopenharmony_ci    if (epollFd < 0) {
80c29fa5a6Sopenharmony_ci        MMI_HILOGE("Invalid param epollFd");
81c29fa5a6Sopenharmony_ci        return RET_ERR;
82c29fa5a6Sopenharmony_ci    }
83c29fa5a6Sopenharmony_ci    auto ret = epoll_wait(epollFd, &events, maxevents, timeout);
84c29fa5a6Sopenharmony_ci    if (ret < 0) {
85c29fa5a6Sopenharmony_ci        MMI_HILOGE("epoll_wait ret:%{public}d,errno:%{public}d", ret, errno);
86c29fa5a6Sopenharmony_ci    }
87c29fa5a6Sopenharmony_ci    return ret;
88c29fa5a6Sopenharmony_ci}
89c29fa5a6Sopenharmony_ci
90c29fa5a6Sopenharmony_civoid UDSSocket::OnReadPackets(CircleStreamBuffer &circBuf, UDSSocket::PacketCallBackFun callbackFun)
91c29fa5a6Sopenharmony_ci{
92c29fa5a6Sopenharmony_ci    constexpr int32_t headSize = static_cast<int32_t>(sizeof(PackHead));
93c29fa5a6Sopenharmony_ci    while (!circBuf.IsEmpty()) {
94c29fa5a6Sopenharmony_ci        const int32_t unreadSize = circBuf.UnreadSize();
95c29fa5a6Sopenharmony_ci        if (unreadSize < headSize) {
96c29fa5a6Sopenharmony_ci            break;
97c29fa5a6Sopenharmony_ci        }
98c29fa5a6Sopenharmony_ci        int32_t dataSize = unreadSize - headSize;
99c29fa5a6Sopenharmony_ci        char *buf = const_cast<char *>(circBuf.ReadBuf());
100c29fa5a6Sopenharmony_ci        CHKPB(buf);
101c29fa5a6Sopenharmony_ci        PackHead *head = reinterpret_cast<PackHead *>(buf);
102c29fa5a6Sopenharmony_ci        CHKPB(head);
103c29fa5a6Sopenharmony_ci        if (head->size < 0 || head->size > MAX_PACKET_BUF_SIZE) {
104c29fa5a6Sopenharmony_ci            MMI_HILOGF("Packet header parsing error, and this error cannot be recovered. The buffer will be reset."
105c29fa5a6Sopenharmony_ci                " head->size:%{public}d, unreadSize:%{public}d", head->size, unreadSize);
106c29fa5a6Sopenharmony_ci            circBuf.Reset();
107c29fa5a6Sopenharmony_ci            break;
108c29fa5a6Sopenharmony_ci        }
109c29fa5a6Sopenharmony_ci        if (head->size > dataSize) {
110c29fa5a6Sopenharmony_ci            break;
111c29fa5a6Sopenharmony_ci        }
112c29fa5a6Sopenharmony_ci        NetPacket pkt(head->idMsg);
113c29fa5a6Sopenharmony_ci        if ((head->size > 0) && (!pkt.Write(&buf[headSize], head->size))) {
114c29fa5a6Sopenharmony_ci            MMI_HILOGW("Error writing data in the NetPacket. It will be retried next time. messageid:%{public}d,"
115c29fa5a6Sopenharmony_ci                "size:%{public}d", head->idMsg, head->size);
116c29fa5a6Sopenharmony_ci            break;
117c29fa5a6Sopenharmony_ci        }
118c29fa5a6Sopenharmony_ci        if (!circBuf.SeekReadPos(pkt.GetPacketLength())) {
119c29fa5a6Sopenharmony_ci            MMI_HILOGW("Set read position error, and this error cannot be recovered, and the buffer will be reset."
120c29fa5a6Sopenharmony_ci                " packetSize:%{public}d unreadSize:%{public}d", pkt.GetPacketLength(), unreadSize);
121c29fa5a6Sopenharmony_ci            circBuf.Reset();
122c29fa5a6Sopenharmony_ci            break;
123c29fa5a6Sopenharmony_ci        }
124c29fa5a6Sopenharmony_ci        callbackFun(pkt);
125c29fa5a6Sopenharmony_ci        if (circBuf.IsEmpty()) {
126c29fa5a6Sopenharmony_ci            circBuf.Reset();
127c29fa5a6Sopenharmony_ci            break;
128c29fa5a6Sopenharmony_ci        }
129c29fa5a6Sopenharmony_ci    }
130c29fa5a6Sopenharmony_ci}
131c29fa5a6Sopenharmony_ci
132c29fa5a6Sopenharmony_civoid UDSSocket::EpollClose()
133c29fa5a6Sopenharmony_ci{
134c29fa5a6Sopenharmony_ci    if (epollFd_ >= 0) {
135c29fa5a6Sopenharmony_ci        close(epollFd_);
136c29fa5a6Sopenharmony_ci        epollFd_ = -1;
137c29fa5a6Sopenharmony_ci    }
138c29fa5a6Sopenharmony_ci}
139c29fa5a6Sopenharmony_ci
140c29fa5a6Sopenharmony_civoid UDSSocket::Close()
141c29fa5a6Sopenharmony_ci{
142c29fa5a6Sopenharmony_ci    if (fd_ >= 0) {
143c29fa5a6Sopenharmony_ci        auto rf = close(fd_);
144c29fa5a6Sopenharmony_ci        if (rf > 0) {
145c29fa5a6Sopenharmony_ci            MMI_HILOGE("Socket close failed rf:%{public}d", rf);
146c29fa5a6Sopenharmony_ci        }
147c29fa5a6Sopenharmony_ci    }
148c29fa5a6Sopenharmony_ci    fd_ = -1;
149c29fa5a6Sopenharmony_ci}
150c29fa5a6Sopenharmony_ci} // namespace MMI
151c29fa5a6Sopenharmony_ci} // namespace OHOS