xref: /aosp_15_r20/frameworks/native/libs/binder/FdUtils.h (revision 38e8c45f13ce32b0dcecb25141ffecaf386fa17f)
1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker  * Copyright (C) 2023 The Android Open Source Project
3*38e8c45fSAndroid Build Coastguard Worker  *
4*38e8c45fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*38e8c45fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*38e8c45fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*38e8c45fSAndroid Build Coastguard Worker  *
8*38e8c45fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*38e8c45fSAndroid Build Coastguard Worker  *
10*38e8c45fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*38e8c45fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*38e8c45fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*38e8c45fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*38e8c45fSAndroid Build Coastguard Worker  * limitations under the License.
15*38e8c45fSAndroid Build Coastguard Worker  */
16*38e8c45fSAndroid Build Coastguard Worker 
17*38e8c45fSAndroid Build Coastguard Worker #pragma once
18*38e8c45fSAndroid Build Coastguard Worker 
19*38e8c45fSAndroid Build Coastguard Worker #include <binder/unique_fd.h>
20*38e8c45fSAndroid Build Coastguard Worker 
21*38e8c45fSAndroid Build Coastguard Worker #if defined(_WIN32) || defined(__TRUSTY__)
22*38e8c45fSAndroid Build Coastguard Worker // Pipe and Socketpair are missing there
23*38e8c45fSAndroid Build Coastguard Worker #elif !defined(BINDER_NO_LIBBASE)
24*38e8c45fSAndroid Build Coastguard Worker 
25*38e8c45fSAndroid Build Coastguard Worker namespace android::binder {
26*38e8c45fSAndroid Build Coastguard Worker using android::base::Pipe;
27*38e8c45fSAndroid Build Coastguard Worker using android::base::Socketpair;
28*38e8c45fSAndroid Build Coastguard Worker } // namespace android::binder
29*38e8c45fSAndroid Build Coastguard Worker 
30*38e8c45fSAndroid Build Coastguard Worker #else // BINDER_NO_LIBBASE
31*38e8c45fSAndroid Build Coastguard Worker 
32*38e8c45fSAndroid Build Coastguard Worker #include <sys/socket.h>
33*38e8c45fSAndroid Build Coastguard Worker 
34*38e8c45fSAndroid Build Coastguard Worker namespace android::binder {
35*38e8c45fSAndroid Build Coastguard Worker 
36*38e8c45fSAndroid Build Coastguard Worker // Inline functions, so that they can be used header-only.
37*38e8c45fSAndroid Build Coastguard Worker 
38*38e8c45fSAndroid Build Coastguard Worker // See pipe(2).
39*38e8c45fSAndroid Build Coastguard Worker // This helper hides the details of converting to unique_fd, and also hides the
40*38e8c45fSAndroid Build Coastguard Worker // fact that macOS doesn't support O_CLOEXEC or O_NONBLOCK directly.
41*38e8c45fSAndroid Build Coastguard Worker inline bool Pipe(unique_fd* read, unique_fd* write, int flags = O_CLOEXEC) {
42*38e8c45fSAndroid Build Coastguard Worker     int pipefd[2];
43*38e8c45fSAndroid Build Coastguard Worker 
44*38e8c45fSAndroid Build Coastguard Worker #if defined(__APPLE__)
45*38e8c45fSAndroid Build Coastguard Worker     if (flags & ~(O_CLOEXEC | O_NONBLOCK)) {
46*38e8c45fSAndroid Build Coastguard Worker         return false;
47*38e8c45fSAndroid Build Coastguard Worker     }
48*38e8c45fSAndroid Build Coastguard Worker     if (pipe(pipefd) != 0) {
49*38e8c45fSAndroid Build Coastguard Worker         return false;
50*38e8c45fSAndroid Build Coastguard Worker     }
51*38e8c45fSAndroid Build Coastguard Worker 
52*38e8c45fSAndroid Build Coastguard Worker     if (flags & O_CLOEXEC) {
53*38e8c45fSAndroid Build Coastguard Worker         if (fcntl(pipefd[0], F_SETFD, FD_CLOEXEC) != 0 ||
54*38e8c45fSAndroid Build Coastguard Worker             fcntl(pipefd[1], F_SETFD, FD_CLOEXEC) != 0) {
55*38e8c45fSAndroid Build Coastguard Worker             close(pipefd[0]);
56*38e8c45fSAndroid Build Coastguard Worker             close(pipefd[1]);
57*38e8c45fSAndroid Build Coastguard Worker             return false;
58*38e8c45fSAndroid Build Coastguard Worker         }
59*38e8c45fSAndroid Build Coastguard Worker     }
60*38e8c45fSAndroid Build Coastguard Worker     if (flags & O_NONBLOCK) {
61*38e8c45fSAndroid Build Coastguard Worker         if (fcntl(pipefd[0], F_SETFL, O_NONBLOCK) != 0 ||
62*38e8c45fSAndroid Build Coastguard Worker             fcntl(pipefd[1], F_SETFL, O_NONBLOCK) != 0) {
63*38e8c45fSAndroid Build Coastguard Worker             close(pipefd[0]);
64*38e8c45fSAndroid Build Coastguard Worker             close(pipefd[1]);
65*38e8c45fSAndroid Build Coastguard Worker             return false;
66*38e8c45fSAndroid Build Coastguard Worker         }
67*38e8c45fSAndroid Build Coastguard Worker     }
68*38e8c45fSAndroid Build Coastguard Worker #else
69*38e8c45fSAndroid Build Coastguard Worker     if (pipe2(pipefd, flags) != 0) {
70*38e8c45fSAndroid Build Coastguard Worker         return false;
71*38e8c45fSAndroid Build Coastguard Worker     }
72*38e8c45fSAndroid Build Coastguard Worker #endif
73*38e8c45fSAndroid Build Coastguard Worker 
74*38e8c45fSAndroid Build Coastguard Worker     read->reset(pipefd[0]);
75*38e8c45fSAndroid Build Coastguard Worker     write->reset(pipefd[1]);
76*38e8c45fSAndroid Build Coastguard Worker     return true;
77*38e8c45fSAndroid Build Coastguard Worker }
78*38e8c45fSAndroid Build Coastguard Worker 
79*38e8c45fSAndroid Build Coastguard Worker // See socketpair(2).
80*38e8c45fSAndroid Build Coastguard Worker // This helper hides the details of converting to unique_fd.
Socketpair(int domain,int type,int protocol,unique_fd * left,unique_fd * right)81*38e8c45fSAndroid Build Coastguard Worker inline bool Socketpair(int domain, int type, int protocol, unique_fd* left, unique_fd* right) {
82*38e8c45fSAndroid Build Coastguard Worker     int sockfd[2];
83*38e8c45fSAndroid Build Coastguard Worker     if (socketpair(domain, type, protocol, sockfd) != 0) {
84*38e8c45fSAndroid Build Coastguard Worker         return false;
85*38e8c45fSAndroid Build Coastguard Worker     }
86*38e8c45fSAndroid Build Coastguard Worker     left->reset(sockfd[0]);
87*38e8c45fSAndroid Build Coastguard Worker     right->reset(sockfd[1]);
88*38e8c45fSAndroid Build Coastguard Worker     return true;
89*38e8c45fSAndroid Build Coastguard Worker }
90*38e8c45fSAndroid Build Coastguard Worker 
91*38e8c45fSAndroid Build Coastguard Worker // See socketpair(2).
92*38e8c45fSAndroid Build Coastguard Worker // This helper hides the details of converting to unique_fd.
Socketpair(int type,unique_fd * left,unique_fd * right)93*38e8c45fSAndroid Build Coastguard Worker inline bool Socketpair(int type, unique_fd* left, unique_fd* right) {
94*38e8c45fSAndroid Build Coastguard Worker     return Socketpair(AF_UNIX, type, 0, left, right);
95*38e8c45fSAndroid Build Coastguard Worker }
96*38e8c45fSAndroid Build Coastguard Worker 
97*38e8c45fSAndroid Build Coastguard Worker } // namespace android::binder
98*38e8c45fSAndroid Build Coastguard Worker 
99*38e8c45fSAndroid Build Coastguard Worker #endif // BINDER_NO_LIBBASE
100