| /* |
| * Copyright (C) 2016 The Android Open Source Project |
| * |
| * 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 <errno.h> |
| #include <string.h> |
| |
| #include "LeakPipe.h" |
| |
| #include "log.h" |
| |
| bool LeakPipe::SendFd(int sock, int fd) { |
| struct msghdr hdr{}; |
| struct iovec iov{}; |
| unsigned int data = 0xfdfdfdfd; |
| alignas(struct cmsghdr) char cmsgbuf[CMSG_SPACE(sizeof(int))]; |
| |
| hdr.msg_iov = &iov; |
| hdr.msg_iovlen = 1; |
| iov.iov_base = &data; |
| iov.iov_len = sizeof(data); |
| |
| hdr.msg_control = cmsgbuf; |
| hdr.msg_controllen = CMSG_LEN(sizeof(int)); |
| |
| struct cmsghdr* cmsg = CMSG_FIRSTHDR(&hdr); |
| cmsg->cmsg_len = CMSG_LEN(sizeof(int)); |
| cmsg->cmsg_level = SOL_SOCKET; |
| cmsg->cmsg_type = SCM_RIGHTS; |
| |
| *(int*)CMSG_DATA(cmsg) = fd; |
| |
| int ret = sendmsg(sock, &hdr, 0); |
| if (ret < 0) { |
| ALOGE("failed to send fd: %s", strerror(errno)); |
| return false; |
| } |
| if (ret == 0) { |
| ALOGE("eof when sending fd"); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| int LeakPipe::ReceiveFd(int sock) { |
| struct msghdr hdr{}; |
| struct iovec iov{}; |
| unsigned int data; |
| alignas(struct cmsghdr) char cmsgbuf[CMSG_SPACE(sizeof(int))]; |
| |
| hdr.msg_iov = &iov; |
| hdr.msg_iovlen = 1; |
| iov.iov_base = &data; |
| iov.iov_len = sizeof(data); |
| |
| hdr.msg_control = cmsgbuf; |
| hdr.msg_controllen = CMSG_LEN(sizeof(int)); |
| |
| int ret = recvmsg(sock, &hdr, 0); |
| if (ret < 0) { |
| ALOGE("failed to receive fd: %s", strerror(errno)); |
| return -1; |
| } |
| if (ret == 0) { |
| ALOGE("eof when receiving fd"); |
| return -1; |
| } |
| |
| struct cmsghdr* cmsg = CMSG_FIRSTHDR(&hdr); |
| if (cmsg == NULL || cmsg->cmsg_level != SOL_SOCKET || cmsg->cmsg_type != SCM_RIGHTS) { |
| ALOGE("missing fd while receiving fd"); |
| return -1; |
| } |
| |
| return *(int*)CMSG_DATA(cmsg); |
| } |