• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2021 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "RpcRawTransport"
18 #include <log/log.h>
19 
20 #include <poll.h>
21 
22 #include <binder/RpcTransportRaw.h>
23 
24 #include "FdTrigger.h"
25 #include "RpcState.h"
26 
27 namespace android {
28 
29 namespace {
30 
31 // RpcTransport with TLS disabled.
32 class RpcTransportRaw : public RpcTransport {
33 public:
RpcTransportRaw(android::base::unique_fd socket)34     explicit RpcTransportRaw(android::base::unique_fd socket) : mSocket(std::move(socket)) {}
peek(void * buf,size_t size,size_t * out_size)35     status_t peek(void* buf, size_t size, size_t* out_size) override {
36         ssize_t ret = TEMP_FAILURE_RETRY(::recv(mSocket.get(), buf, size, MSG_PEEK));
37         if (ret < 0) {
38             int savedErrno = errno;
39             if (savedErrno == EAGAIN || savedErrno == EWOULDBLOCK) {
40                 return WOULD_BLOCK;
41             }
42 
43             LOG_RPC_DETAIL("RpcTransport peek(): %s", strerror(savedErrno));
44             return -savedErrno;
45         }
46 
47         *out_size = static_cast<size_t>(ret);
48         return OK;
49     }
50 
51     template <typename SendOrReceive>
interruptableReadOrWrite(FdTrigger * fdTrigger,iovec * iovs,int niovs,SendOrReceive sendOrReceiveFun,const char * funName,int16_t event,const std::function<status_t ()> & altPoll)52     status_t interruptableReadOrWrite(FdTrigger* fdTrigger, iovec* iovs, int niovs,
53                                       SendOrReceive sendOrReceiveFun, const char* funName,
54                                       int16_t event, const std::function<status_t()>& altPoll) {
55         MAYBE_WAIT_IN_FLAKE_MODE;
56 
57         if (niovs < 0) {
58             return BAD_VALUE;
59         }
60 
61         // Since we didn't poll, we need to manually check to see if it was triggered. Otherwise, we
62         // may never know we should be shutting down.
63         if (fdTrigger->isTriggered()) {
64             return DEAD_OBJECT;
65         }
66 
67         // If iovs has one or more empty vectors at the end and
68         // we somehow advance past all the preceding vectors and
69         // pass some or all of the empty ones to sendmsg/recvmsg,
70         // the call will return processSize == 0. In that case
71         // we should be returning OK but instead return DEAD_OBJECT.
72         // To avoid this problem, we make sure here that the last
73         // vector at iovs[niovs - 1] has a non-zero length.
74         while (niovs > 0 && iovs[niovs - 1].iov_len == 0) {
75             niovs--;
76         }
77         if (niovs == 0) {
78             // The vectors are all empty, so we have nothing to send.
79             return OK;
80         }
81 
82         bool havePolled = false;
83         while (true) {
84             msghdr msg{
85                     .msg_iov = iovs,
86                     // posix uses int, glibc uses size_t.  niovs is a
87                     // non-negative int and can be cast to either.
88                     .msg_iovlen = static_cast<decltype(msg.msg_iovlen)>(niovs),
89             };
90             ssize_t processSize =
91                     TEMP_FAILURE_RETRY(sendOrReceiveFun(mSocket.get(), &msg, MSG_NOSIGNAL));
92 
93             if (processSize < 0) {
94                 int savedErrno = errno;
95 
96                 // Still return the error on later passes, since it would expose
97                 // a problem with polling
98                 if (havePolled || (savedErrno != EAGAIN && savedErrno != EWOULDBLOCK)) {
99                     LOG_RPC_DETAIL("RpcTransport %s(): %s", funName, strerror(savedErrno));
100                     return -savedErrno;
101                 }
102             } else if (processSize == 0) {
103                 return DEAD_OBJECT;
104             } else {
105                 while (processSize > 0 && niovs > 0) {
106                     auto& iov = iovs[0];
107                     if (static_cast<size_t>(processSize) < iov.iov_len) {
108                         // Advance the base of the current iovec
109                         iov.iov_base = reinterpret_cast<char*>(iov.iov_base) + processSize;
110                         iov.iov_len -= processSize;
111                         break;
112                     }
113 
114                     // The current iovec was fully written
115                     processSize -= iov.iov_len;
116                     iovs++;
117                     niovs--;
118                 }
119                 if (niovs == 0) {
120                     LOG_ALWAYS_FATAL_IF(processSize > 0,
121                                         "Reached the end of iovecs "
122                                         "with %zd bytes remaining",
123                                         processSize);
124                     return OK;
125                 }
126             }
127 
128             if (altPoll) {
129                 if (status_t status = altPoll(); status != OK) return status;
130                 if (fdTrigger->isTriggered()) {
131                     return DEAD_OBJECT;
132                 }
133             } else {
134                 if (status_t status = fdTrigger->triggerablePoll(mSocket.get(), event);
135                     status != OK)
136                     return status;
137                 if (!havePolled) havePolled = true;
138             }
139         }
140     }
141 
interruptableWriteFully(FdTrigger * fdTrigger,iovec * iovs,int niovs,const std::function<status_t ()> & altPoll)142     status_t interruptableWriteFully(FdTrigger* fdTrigger, iovec* iovs, int niovs,
143                                      const std::function<status_t()>& altPoll) override {
144         return interruptableReadOrWrite(fdTrigger, iovs, niovs, sendmsg, "sendmsg", POLLOUT,
145                                         altPoll);
146     }
147 
interruptableReadFully(FdTrigger * fdTrigger,iovec * iovs,int niovs,const std::function<status_t ()> & altPoll)148     status_t interruptableReadFully(FdTrigger* fdTrigger, iovec* iovs, int niovs,
149                                     const std::function<status_t()>& altPoll) override {
150         return interruptableReadOrWrite(fdTrigger, iovs, niovs, recvmsg, "recvmsg", POLLIN,
151                                         altPoll);
152     }
153 
154 private:
155     base::unique_fd mSocket;
156 };
157 
158 // RpcTransportCtx with TLS disabled.
159 class RpcTransportCtxRaw : public RpcTransportCtx {
160 public:
newTransport(android::base::unique_fd fd,FdTrigger *) const161     std::unique_ptr<RpcTransport> newTransport(android::base::unique_fd fd, FdTrigger*) const {
162         return std::make_unique<RpcTransportRaw>(std::move(fd));
163     }
getCertificate(RpcCertificateFormat) const164     std::vector<uint8_t> getCertificate(RpcCertificateFormat) const override { return {}; }
165 };
166 
167 } // namespace
168 
newServerCtx() const169 std::unique_ptr<RpcTransportCtx> RpcTransportCtxFactoryRaw::newServerCtx() const {
170     return std::make_unique<RpcTransportCtxRaw>();
171 }
172 
newClientCtx() const173 std::unique_ptr<RpcTransportCtx> RpcTransportCtxFactoryRaw::newClientCtx() const {
174     return std::make_unique<RpcTransportCtxRaw>();
175 }
176 
toCString() const177 const char *RpcTransportCtxFactoryRaw::toCString() const {
178     return "raw";
179 }
180 
make()181 std::unique_ptr<RpcTransportCtxFactory> RpcTransportCtxFactoryRaw::make() {
182     return std::unique_ptr<RpcTransportCtxFactoryRaw>(new RpcTransportCtxFactoryRaw());
183 }
184 
185 } // namespace android
186