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