/* * Copyright (c) 2021-2022 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "uds_socket.h" #include #include "error_multimodal.h" #include "mmi_log.h" #undef MMI_LOG_TAG #define MMI_LOG_TAG "UDSSocket" namespace OHOS { namespace MMI { UDSSocket::UDSSocket() {} UDSSocket::~UDSSocket() { Close(); EpollClose(); } int32_t UDSSocket::EpollCreate(int32_t size) { epollFd_ = epoll_create(size); if (epollFd_ < 0) { MMI_HILOGE("epoll_create return %{public}d", epollFd_); } else { MMI_HILOGI("epoll_create, epollFd_:%{public}d", epollFd_); } return epollFd_; } int32_t UDSSocket::EpollCtl(int32_t fd, int32_t op, struct epoll_event &event, int32_t epollFd) { if (fd < 0) { MMI_HILOGE("Invalid fd"); return RET_ERR; } if (epollFd < 0) { epollFd = epollFd_; } if (epollFd < 0) { MMI_HILOGE("Invalid param epollFd"); return RET_ERR; } int32_t ret; if (op == EPOLL_CTL_DEL) { ret = epoll_ctl(epollFd, op, fd, nullptr); } else { ret = epoll_ctl(epollFd, op, fd, &event); } if (ret < 0) { MMI_HILOGE("epoll_ctl return %{public}d,epollFd_:%{public}d," "op:%{public}d,fd:%{public}d,errno:%{public}d", ret, epollFd, op, fd, errno); } return ret; } int32_t UDSSocket::EpollWait(struct epoll_event &events, int32_t maxevents, int32_t timeout, int32_t epollFd) { if (epollFd < 0) { epollFd = epollFd_; } if (epollFd < 0) { MMI_HILOGE("Invalid param epollFd"); return RET_ERR; } auto ret = epoll_wait(epollFd, &events, maxevents, timeout); if (ret < 0) { MMI_HILOGE("epoll_wait ret:%{public}d,errno:%{public}d", ret, errno); } return ret; } void UDSSocket::OnReadPackets(CircleStreamBuffer &circBuf, UDSSocket::PacketCallBackFun callbackFun) { constexpr int32_t headSize = static_cast(sizeof(PackHead)); while (!circBuf.IsEmpty()) { const int32_t unreadSize = circBuf.UnreadSize(); if (unreadSize < headSize) { break; } int32_t dataSize = unreadSize - headSize; char *buf = const_cast(circBuf.ReadBuf()); CHKPB(buf); PackHead *head = reinterpret_cast(buf); CHKPB(head); if (head->size < 0 || head->size > MAX_PACKET_BUF_SIZE) { MMI_HILOGF("Packet header parsing error, and this error cannot be recovered. The buffer will be reset." " head->size:%{public}d, unreadSize:%{public}d", head->size, unreadSize); circBuf.Reset(); break; } if (head->size > dataSize) { break; } NetPacket pkt(head->idMsg); if ((head->size > 0) && (!pkt.Write(&buf[headSize], head->size))) { MMI_HILOGW("Error writing data in the NetPacket. It will be retried next time. messageid:%{public}d," "size:%{public}d", head->idMsg, head->size); break; } if (!circBuf.SeekReadPos(pkt.GetPacketLength())) { MMI_HILOGW("Set read position error, and this error cannot be recovered, and the buffer will be reset." " packetSize:%{public}d unreadSize:%{public}d", pkt.GetPacketLength(), unreadSize); circBuf.Reset(); break; } callbackFun(pkt); if (circBuf.IsEmpty()) { circBuf.Reset(); break; } } } void UDSSocket::EpollClose() { if (epollFd_ >= 0) { close(epollFd_); epollFd_ = -1; } } void UDSSocket::Close() { if (fd_ >= 0) { auto rf = close(fd_); if (rf > 0) { MMI_HILOGE("Socket close failed rf:%{public}d", rf); } } fd_ = -1; } } // namespace MMI } // namespace OHOS