[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 1 | // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "net/socket/unix_domain_server_socket_posix.h" |
| 6 | |
| 7 | #include <errno.h> |
| 8 | #include <sys/socket.h> |
| 9 | #include <sys/un.h> |
| 10 | #include <unistd.h> |
dcheng | c7eeda42 | 2015-12-26 03:56:48 | [diff] [blame] | 11 | #include <utility> |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 12 | |
| 13 | #include "base/logging.h" |
Sergey Ulanov | 4908557 | 2017-07-10 23:25:46 | [diff] [blame] | 14 | #include "build/build_config.h" |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 15 | #include "net/base/net_errors.h" |
tfarina | 3d87d7cd | 2016-01-13 02:26:59 | [diff] [blame] | 16 | #include "net/base/sockaddr_storage.h" |
tfarina | 4eb7aad8 | 2015-09-14 17:10:34 | [diff] [blame] | 17 | #include "net/socket/socket_posix.h" |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 18 | #include "net/socket/unix_domain_client_socket_posix.h" |
| 19 | |
| 20 | namespace net { |
| 21 | |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 22 | namespace { |
| 23 | |
| 24 | // Intended for use as SetterCallbacks in Accept() helper methods. |
danakj | 655b66c | 2016-04-16 00:51:38 | [diff] [blame] | 25 | void SetStreamSocket(std::unique_ptr<StreamSocket>* socket, |
| 26 | std::unique_ptr<SocketPosix> accepted_socket) { |
dcheng | c7eeda42 | 2015-12-26 03:56:48 | [diff] [blame] | 27 | socket->reset(new UnixDomainClientSocket(std::move(accepted_socket))); |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 28 | } |
| 29 | |
| 30 | void SetSocketDescriptor(SocketDescriptor* socket, |
danakj | 655b66c | 2016-04-16 00:51:38 | [diff] [blame] | 31 | std::unique_ptr<SocketPosix> accepted_socket) { |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 32 | *socket = accepted_socket->ReleaseConnectedSocket(); |
| 33 | } |
| 34 | |
| 35 | } // anonymous namespace |
| 36 | |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 37 | UnixDomainServerSocket::UnixDomainServerSocket( |
| 38 | const AuthCallback& auth_callback, |
| 39 | bool use_abstract_namespace) |
| 40 | : auth_callback_(auth_callback), |
| 41 | use_abstract_namespace_(use_abstract_namespace) { |
| 42 | DCHECK(!auth_callback_.is_null()); |
| 43 | } |
| 44 | |
Chris Watkins | 7a41d355 | 2017-12-01 02:13:27 | [diff] [blame] | 45 | UnixDomainServerSocket::~UnixDomainServerSocket() = default; |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 46 | |
| 47 | // static |
[email protected] | fe928be9 | 2014-08-08 09:17:41 | [diff] [blame] | 48 | bool UnixDomainServerSocket::GetPeerCredentials(SocketDescriptor socket, |
| 49 | Credentials* credentials) { |
Sergey Ulanov | 4908557 | 2017-07-10 23:25:46 | [diff] [blame] | 50 | #if defined(OS_LINUX) || defined(OS_ANDROID) || defined(OS_FUCHSIA) |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 51 | struct ucred user_cred; |
| 52 | socklen_t len = sizeof(user_cred); |
| 53 | if (getsockopt(socket, SOL_SOCKET, SO_PEERCRED, &user_cred, &len) < 0) |
| 54 | return false; |
[email protected] | fe928be9 | 2014-08-08 09:17:41 | [diff] [blame] | 55 | credentials->process_id = user_cred.pid; |
| 56 | credentials->user_id = user_cred.uid; |
| 57 | credentials->group_id = user_cred.gid; |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 58 | return true; |
| 59 | #else |
[email protected] | fe928be9 | 2014-08-08 09:17:41 | [diff] [blame] | 60 | return getpeereid( |
| 61 | socket, &credentials->user_id, &credentials->group_id) == 0; |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 62 | #endif |
| 63 | } |
| 64 | |
| 65 | int UnixDomainServerSocket::Listen(const IPEndPoint& address, int backlog) { |
| 66 | NOTIMPLEMENTED(); |
| 67 | return ERR_NOT_IMPLEMENTED; |
| 68 | } |
| 69 | |
bcf | 5cbaa77 | 2016-02-24 02:22:30 | [diff] [blame] | 70 | int UnixDomainServerSocket::ListenWithAddressAndPort( |
| 71 | const std::string& address_string, |
| 72 | uint16_t port, |
| 73 | int backlog) { |
| 74 | NOTIMPLEMENTED(); |
| 75 | return ERR_NOT_IMPLEMENTED; |
| 76 | } |
| 77 | |
tfarina | a7b245d | 2016-02-02 02:03:49 | [diff] [blame] | 78 | int UnixDomainServerSocket::BindAndListen(const std::string& socket_path, |
| 79 | int backlog) { |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 80 | DCHECK(!listen_socket_); |
| 81 | |
| 82 | SockaddrStorage address; |
tfarina | a7b245d | 2016-02-02 02:03:49 | [diff] [blame] | 83 | if (!UnixDomainClientSocket::FillAddress(socket_path, |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 84 | use_abstract_namespace_, |
| 85 | &address)) { |
| 86 | return ERR_ADDRESS_INVALID; |
| 87 | } |
| 88 | |
danakj | 655b66c | 2016-04-16 00:51:38 | [diff] [blame] | 89 | std::unique_ptr<SocketPosix> socket(new SocketPosix); |
byungchul | 14ac248 | 2014-08-27 23:04:26 | [diff] [blame] | 90 | int rv = socket->Open(AF_UNIX); |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 91 | DCHECK_NE(ERR_IO_PENDING, rv); |
| 92 | if (rv != OK) |
| 93 | return rv; |
| 94 | |
byungchul | 14ac248 | 2014-08-27 23:04:26 | [diff] [blame] | 95 | rv = socket->Bind(address); |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 96 | DCHECK_NE(ERR_IO_PENDING, rv); |
| 97 | if (rv != OK) { |
| 98 | PLOG(ERROR) |
tfarina | a7b245d | 2016-02-02 02:03:49 | [diff] [blame] | 99 | << "Could not bind unix domain socket to " << socket_path |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 100 | << (use_abstract_namespace_ ? " (with abstract namespace)" : ""); |
| 101 | return rv; |
| 102 | } |
| 103 | |
byungchul | 14ac248 | 2014-08-27 23:04:26 | [diff] [blame] | 104 | rv = socket->Listen(backlog); |
| 105 | DCHECK_NE(ERR_IO_PENDING, rv); |
| 106 | if (rv != OK) |
| 107 | return rv; |
| 108 | |
| 109 | listen_socket_.swap(socket); |
| 110 | return rv; |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 111 | } |
| 112 | |
| 113 | int UnixDomainServerSocket::GetLocalAddress(IPEndPoint* address) const { |
tobiasjs | 72395bd | 2015-07-06 10:06:53 | [diff] [blame] | 114 | DCHECK(address); |
| 115 | |
| 116 | // Unix domain sockets have no valid associated addr/port; |
| 117 | // return address invalid. |
| 118 | return ERR_ADDRESS_INVALID; |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 119 | } |
| 120 | |
danakj | 655b66c | 2016-04-16 00:51:38 | [diff] [blame] | 121 | int UnixDomainServerSocket::Accept(std::unique_ptr<StreamSocket>* socket, |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 122 | const CompletionCallback& callback) { |
| 123 | DCHECK(socket); |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 124 | |
| 125 | SetterCallback setter_callback = base::Bind(&SetStreamSocket, socket); |
| 126 | return DoAccept(setter_callback, callback); |
| 127 | } |
| 128 | |
| 129 | int UnixDomainServerSocket::AcceptSocketDescriptor( |
| 130 | SocketDescriptor* socket, |
| 131 | const CompletionCallback& callback) { |
| 132 | DCHECK(socket); |
| 133 | |
| 134 | SetterCallback setter_callback = base::Bind(&SetSocketDescriptor, socket); |
| 135 | return DoAccept(setter_callback, callback); |
| 136 | } |
| 137 | |
| 138 | int UnixDomainServerSocket::DoAccept(const SetterCallback& setter_callback, |
| 139 | const CompletionCallback& callback) { |
| 140 | DCHECK(!setter_callback.is_null()); |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 141 | DCHECK(!callback.is_null()); |
| 142 | DCHECK(listen_socket_); |
| 143 | DCHECK(!accept_socket_); |
| 144 | |
| 145 | while (true) { |
| 146 | int rv = listen_socket_->Accept( |
| 147 | &accept_socket_, |
| 148 | base::Bind(&UnixDomainServerSocket::AcceptCompleted, |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 149 | base::Unretained(this), |
| 150 | setter_callback, |
| 151 | callback)); |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 152 | if (rv != OK) |
| 153 | return rv; |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 154 | if (AuthenticateAndGetStreamSocket(setter_callback)) |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 155 | return OK; |
| 156 | // Accept another socket because authentication error should be transparent |
| 157 | // to the caller. |
| 158 | } |
| 159 | } |
| 160 | |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 161 | void UnixDomainServerSocket::AcceptCompleted( |
| 162 | const SetterCallback& setter_callback, |
| 163 | const CompletionCallback& callback, |
| 164 | int rv) { |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 165 | if (rv != OK) { |
| 166 | callback.Run(rv); |
| 167 | return; |
| 168 | } |
| 169 | |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 170 | if (AuthenticateAndGetStreamSocket(setter_callback)) { |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 171 | callback.Run(OK); |
| 172 | return; |
| 173 | } |
| 174 | |
| 175 | // Accept another socket because authentication error should be transparent |
| 176 | // to the caller. |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 177 | rv = DoAccept(setter_callback, callback); |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 178 | if (rv != ERR_IO_PENDING) |
| 179 | callback.Run(rv); |
| 180 | } |
| 181 | |
| 182 | bool UnixDomainServerSocket::AuthenticateAndGetStreamSocket( |
cmasone | ca100d5 | 2014-09-03 18:11:11 | [diff] [blame] | 183 | const SetterCallback& setter_callback) { |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 184 | DCHECK(accept_socket_); |
| 185 | |
[email protected] | fe928be9 | 2014-08-08 09:17:41 | [diff] [blame] | 186 | Credentials credentials; |
| 187 | if (!GetPeerCredentials(accept_socket_->socket_fd(), &credentials) || |
| 188 | !auth_callback_.Run(credentials)) { |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 189 | accept_socket_.reset(); |
| 190 | return false; |
| 191 | } |
| 192 | |
dcheng | c7eeda42 | 2015-12-26 03:56:48 | [diff] [blame] | 193 | setter_callback.Run(std::move(accept_socket_)); |
[email protected] | 518c63a | 2014-07-24 03:51:23 | [diff] [blame] | 194 | return true; |
| 195 | } |
| 196 | |
| 197 | } // namespace net |