blob: f5cc41332e192ce95c8bd01e8c4ad2eca55f0868 [file] [log] [blame]
Yifan Hongf6b4d5c2021-06-23 19:12:32 -07001/*
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
Yifan Hong8c950422021-08-05 17:13:55 -070020#include <poll.h>
21
Yifan Hongf6b4d5c2021-06-23 19:12:32 -070022#include <binder/RpcTransportRaw.h>
23
Yifan Hong8c950422021-08-05 17:13:55 -070024#include "FdTrigger.h"
Yifan Hongf6b4d5c2021-06-23 19:12:32 -070025#include "RpcState.h"
26
Yifan Hongf6b4d5c2021-06-23 19:12:32 -070027namespace android {
28
29namespace {
30
31// RpcTransport with TLS disabled.
32class RpcTransportRaw : public RpcTransport {
33public:
34 explicit RpcTransportRaw(android::base::unique_fd socket) : mSocket(std::move(socket)) {}
Andrei Homescu1975aaa2022-03-19 02:34:57 +000035 status_t pollRead(void) override {
36 uint8_t buf;
37 ssize_t ret = TEMP_FAILURE_RETRY(
38 ::recv(mSocket.get(), &buf, sizeof(buf), MSG_PEEK | MSG_DONTWAIT));
Yifan Hongf6b4d5c2021-06-23 19:12:32 -070039 if (ret < 0) {
Andrei Homescu5ad71b52022-03-11 03:49:12 +000040 int savedErrno = errno;
41 if (savedErrno == EAGAIN || savedErrno == EWOULDBLOCK) {
42 return WOULD_BLOCK;
43 }
44
Andrei Homescu1975aaa2022-03-19 02:34:57 +000045 LOG_RPC_DETAIL("RpcTransport poll(): %s", strerror(savedErrno));
Andrei Homescu5ad71b52022-03-11 03:49:12 +000046 return -savedErrno;
Andrei Homescu1975aaa2022-03-19 02:34:57 +000047 } else if (ret == 0) {
48 return DEAD_OBJECT;
Yifan Hongf6b4d5c2021-06-23 19:12:32 -070049 }
Andrei Homescu5ad71b52022-03-11 03:49:12 +000050
Andrei Homescu5ad71b52022-03-11 03:49:12 +000051 return OK;
Yifan Hongf6b4d5c2021-06-23 19:12:32 -070052 }
Yifan Hong8c950422021-08-05 17:13:55 -070053
Andrei Homescua39e4ed2021-12-10 08:41:54 +000054 template <typename SendOrReceive>
Colin Cross9adfeaf2022-01-21 17:22:09 -080055 status_t interruptableReadOrWrite(FdTrigger* fdTrigger, iovec* iovs, int niovs,
Steven Moreland301c3f02021-09-14 17:49:04 -070056 SendOrReceive sendOrReceiveFun, const char* funName,
Steven Moreland43921d52021-09-27 17:15:56 -070057 int16_t event, const std::function<status_t()>& altPoll) {
Yifan Hong8c950422021-08-05 17:13:55 -070058 MAYBE_WAIT_IN_FLAKE_MODE;
59
Colin Cross9adfeaf2022-01-21 17:22:09 -080060 if (niovs < 0) {
61 return BAD_VALUE;
62 }
63
Steven Moreland301c3f02021-09-14 17:49:04 -070064 // Since we didn't poll, we need to manually check to see if it was triggered. Otherwise, we
65 // may never know we should be shutting down.
66 if (fdTrigger->isTriggered()) {
67 return DEAD_OBJECT;
68 }
69
Andrei Homescua39e4ed2021-12-10 08:41:54 +000070 // If iovs has one or more empty vectors at the end and
71 // we somehow advance past all the preceding vectors and
72 // pass some or all of the empty ones to sendmsg/recvmsg,
73 // the call will return processSize == 0. In that case
74 // we should be returning OK but instead return DEAD_OBJECT.
75 // To avoid this problem, we make sure here that the last
76 // vector at iovs[niovs - 1] has a non-zero length.
77 while (niovs > 0 && iovs[niovs - 1].iov_len == 0) {
78 niovs--;
79 }
80 if (niovs == 0) {
81 // The vectors are all empty, so we have nothing to send.
82 return OK;
83 }
84
Steven Moreland43921d52021-09-27 17:15:56 -070085 bool havePolled = false;
86 while (true) {
Andrei Homescua39e4ed2021-12-10 08:41:54 +000087 msghdr msg{
88 .msg_iov = iovs,
Colin Cross9adfeaf2022-01-21 17:22:09 -080089 // posix uses int, glibc uses size_t. niovs is a
90 // non-negative int and can be cast to either.
91 .msg_iovlen = static_cast<decltype(msg.msg_iovlen)>(niovs),
Andrei Homescua39e4ed2021-12-10 08:41:54 +000092 };
93 ssize_t processSize =
94 TEMP_FAILURE_RETRY(sendOrReceiveFun(mSocket.get(), &msg, MSG_NOSIGNAL));
Steven Moreland301c3f02021-09-14 17:49:04 -070095
96 if (processSize < 0) {
Steven Moreland5252e752021-09-14 14:06:03 -070097 int savedErrno = errno;
Steven Moreland301c3f02021-09-14 17:49:04 -070098
99 // Still return the error on later passes, since it would expose
100 // a problem with polling
Andrei Homescua39e4ed2021-12-10 08:41:54 +0000101 if (havePolled || (savedErrno != EAGAIN && savedErrno != EWOULDBLOCK)) {
Steven Moreland301c3f02021-09-14 17:49:04 -0700102 LOG_RPC_DETAIL("RpcTransport %s(): %s", funName, strerror(savedErrno));
103 return -savedErrno;
104 }
105 } else if (processSize == 0) {
106 return DEAD_OBJECT;
107 } else {
Andrei Homescua39e4ed2021-12-10 08:41:54 +0000108 while (processSize > 0 && niovs > 0) {
109 auto& iov = iovs[0];
110 if (static_cast<size_t>(processSize) < iov.iov_len) {
111 // Advance the base of the current iovec
112 iov.iov_base = reinterpret_cast<char*>(iov.iov_base) + processSize;
113 iov.iov_len -= processSize;
114 break;
115 }
116
117 // The current iovec was fully written
118 processSize -= iov.iov_len;
119 iovs++;
120 niovs--;
121 }
122 if (niovs == 0) {
123 LOG_ALWAYS_FATAL_IF(processSize > 0,
124 "Reached the end of iovecs "
125 "with %zd bytes remaining",
126 processSize);
Steven Moreland301c3f02021-09-14 17:49:04 -0700127 return OK;
128 }
Yifan Hong8c950422021-08-05 17:13:55 -0700129 }
130
Steven Moreland43921d52021-09-27 17:15:56 -0700131 if (altPoll) {
132 if (status_t status = altPoll(); status != OK) return status;
133 if (fdTrigger->isTriggered()) {
134 return DEAD_OBJECT;
135 }
136 } else {
137 if (status_t status = fdTrigger->triggerablePoll(mSocket.get(), event);
138 status != OK)
139 return status;
140 if (!havePolled) havePolled = true;
141 }
142 }
Yifan Hong8c950422021-08-05 17:13:55 -0700143 }
144
Colin Cross9adfeaf2022-01-21 17:22:09 -0800145 status_t interruptableWriteFully(FdTrigger* fdTrigger, iovec* iovs, int niovs,
Steven Moreland43921d52021-09-27 17:15:56 -0700146 const std::function<status_t()>& altPoll) override {
Andrei Homescua39e4ed2021-12-10 08:41:54 +0000147 return interruptableReadOrWrite(fdTrigger, iovs, niovs, sendmsg, "sendmsg", POLLOUT,
148 altPoll);
Steven Moreland301c3f02021-09-14 17:49:04 -0700149 }
150
Colin Cross9adfeaf2022-01-21 17:22:09 -0800151 status_t interruptableReadFully(FdTrigger* fdTrigger, iovec* iovs, int niovs,
Steven Moreland43921d52021-09-27 17:15:56 -0700152 const std::function<status_t()>& altPoll) override {
Andrei Homescua39e4ed2021-12-10 08:41:54 +0000153 return interruptableReadOrWrite(fdTrigger, iovs, niovs, recvmsg, "recvmsg", POLLIN,
154 altPoll);
Yifan Hong8c950422021-08-05 17:13:55 -0700155 }
Yifan Hongf6b4d5c2021-06-23 19:12:32 -0700156
157private:
Steven Moreland301c3f02021-09-14 17:49:04 -0700158 base::unique_fd mSocket;
Yifan Hongf6b4d5c2021-06-23 19:12:32 -0700159};
160
161// RpcTransportCtx with TLS disabled.
162class RpcTransportCtxRaw : public RpcTransportCtx {
163public:
Yifan Hongf6d42292021-08-05 23:43:05 -0700164 std::unique_ptr<RpcTransport> newTransport(android::base::unique_fd fd, FdTrigger*) const {
Yifan Hongf6b4d5c2021-06-23 19:12:32 -0700165 return std::make_unique<RpcTransportRaw>(std::move(fd));
166 }
Yifan Hong9734cfc2021-09-13 16:14:09 -0700167 std::vector<uint8_t> getCertificate(RpcCertificateFormat) const override { return {}; }
Yifan Hongf6b4d5c2021-06-23 19:12:32 -0700168};
Yifan Hong588d59c2021-08-16 17:13:58 -0700169
Yifan Hongf6b4d5c2021-06-23 19:12:32 -0700170} // namespace
171
172std::unique_ptr<RpcTransportCtx> RpcTransportCtxFactoryRaw::newServerCtx() const {
173 return std::make_unique<RpcTransportCtxRaw>();
174}
175
176std::unique_ptr<RpcTransportCtx> RpcTransportCtxFactoryRaw::newClientCtx() const {
177 return std::make_unique<RpcTransportCtxRaw>();
178}
179
180const char *RpcTransportCtxFactoryRaw::toCString() const {
181 return "raw";
182}
183
184std::unique_ptr<RpcTransportCtxFactory> RpcTransportCtxFactoryRaw::make() {
185 return std::unique_ptr<RpcTransportCtxFactoryRaw>(new RpcTransportCtxFactoryRaw());
186}
187
188} // namespace android