1 /* 2 * Copyright (C) 2007 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 package android.net; 18 19 import java.io.FileDescriptor; 20 import java.io.IOException; 21 import java.io.InputStream; 22 import java.io.OutputStream; 23 import java.net.SocketOptions; 24 25 /** 26 * Creates a (non-server) socket in the UNIX-domain namespace. The interface 27 * here is not entirely unlike that of java.net.Socket 28 */ 29 public class LocalSocket { 30 31 private LocalSocketImpl impl; 32 private volatile boolean implCreated; 33 private LocalSocketAddress localAddress; 34 private boolean isBound; 35 private boolean isConnected; 36 37 /** 38 * Creates a AF_LOCAL/UNIX domain stream socket. 39 */ LocalSocket()40 public LocalSocket() { 41 this(new LocalSocketImpl()); 42 isBound = false; 43 isConnected = false; 44 } 45 46 /** 47 * for use with AndroidServerSocket 48 * @param impl a SocketImpl 49 */ LocalSocket(LocalSocketImpl impl)50 /*package*/ LocalSocket(LocalSocketImpl impl) { 51 this.impl = impl; 52 this.isConnected = false; 53 this.isBound = false; 54 } 55 56 /** {@inheritDoc} */ 57 @Override toString()58 public String toString() { 59 return super.toString() + " impl:" + impl; 60 } 61 62 /** 63 * It's difficult to discern from the spec when impl.create() should be 64 * called, but it seems like a reasonable rule is "as soon as possible, 65 * but not in a context where IOException cannot be thrown" 66 * 67 * @throws IOException from SocketImpl.create() 68 */ implCreateIfNeeded()69 private void implCreateIfNeeded() throws IOException { 70 if (!implCreated) { 71 synchronized (this) { 72 if (!implCreated) { 73 implCreated = true; 74 impl.create(true); 75 } 76 } 77 } 78 } 79 80 /** 81 * Connects this socket to an endpoint. May only be called on an instance 82 * that has not yet been connected. 83 * 84 * @param endpoint endpoint address 85 * @throws IOException if socket is in invalid state or the address does 86 * not exist. 87 */ connect(LocalSocketAddress endpoint)88 public void connect(LocalSocketAddress endpoint) throws IOException { 89 synchronized (this) { 90 if (isConnected) { 91 throw new IOException("already connected"); 92 } 93 94 implCreateIfNeeded(); 95 impl.connect(endpoint, 0); 96 isConnected = true; 97 isBound = true; 98 } 99 } 100 101 /** 102 * Binds this socket to an endpoint name. May only be called on an instance 103 * that has not yet been bound. 104 * 105 * @param bindpoint endpoint address 106 * @throws IOException 107 */ bind(LocalSocketAddress bindpoint)108 public void bind(LocalSocketAddress bindpoint) throws IOException { 109 implCreateIfNeeded(); 110 111 synchronized (this) { 112 if (isBound) { 113 throw new IOException("already bound"); 114 } 115 116 localAddress = bindpoint; 117 impl.bind(localAddress); 118 isBound = true; 119 } 120 } 121 122 /** 123 * Retrieves the name that this socket is bound to, if any. 124 * 125 * @return Local address or null if anonymous 126 */ getLocalSocketAddress()127 public LocalSocketAddress getLocalSocketAddress() { 128 return localAddress; 129 } 130 131 /** 132 * Retrieves the input stream for this instance. 133 * 134 * @return input stream 135 * @throws IOException if socket has been closed or cannot be created. 136 */ getInputStream()137 public InputStream getInputStream() throws IOException { 138 implCreateIfNeeded(); 139 return impl.getInputStream(); 140 } 141 142 /** 143 * Retrieves the output stream for this instance. 144 * 145 * @return output stream 146 * @throws IOException if socket has been closed or cannot be created. 147 */ getOutputStream()148 public OutputStream getOutputStream() throws IOException { 149 implCreateIfNeeded(); 150 return impl.getOutputStream(); 151 } 152 153 /** 154 * Closes the socket. 155 * 156 * @throws IOException 157 */ close()158 public void close() throws IOException { 159 implCreateIfNeeded(); 160 impl.close(); 161 } 162 163 /** 164 * Shuts down the input side of the socket. 165 * 166 * @throws IOException 167 */ shutdownInput()168 public void shutdownInput() throws IOException { 169 implCreateIfNeeded(); 170 impl.shutdownInput(); 171 } 172 173 /** 174 * Shuts down the output side of the socket. 175 * 176 * @throws IOException 177 */ shutdownOutput()178 public void shutdownOutput() throws IOException { 179 implCreateIfNeeded(); 180 impl.shutdownOutput(); 181 } 182 setReceiveBufferSize(int size)183 public void setReceiveBufferSize(int size) throws IOException { 184 impl.setOption(SocketOptions.SO_RCVBUF, Integer.valueOf(size)); 185 } 186 getReceiveBufferSize()187 public int getReceiveBufferSize() throws IOException { 188 return ((Integer) impl.getOption(SocketOptions.SO_RCVBUF)).intValue(); 189 } 190 setSoTimeout(int n)191 public void setSoTimeout(int n) throws IOException { 192 impl.setOption(SocketOptions.SO_TIMEOUT, Integer.valueOf(n)); 193 } 194 getSoTimeout()195 public int getSoTimeout() throws IOException { 196 return ((Integer) impl.getOption(SocketOptions.SO_TIMEOUT)).intValue(); 197 } 198 setSendBufferSize(int n)199 public void setSendBufferSize(int n) throws IOException { 200 impl.setOption(SocketOptions.SO_SNDBUF, Integer.valueOf(n)); 201 } 202 getSendBufferSize()203 public int getSendBufferSize() throws IOException { 204 return ((Integer) impl.getOption(SocketOptions.SO_SNDBUF)).intValue(); 205 } 206 207 //???SEC getRemoteSocketAddress()208 public LocalSocketAddress getRemoteSocketAddress() { 209 throw new UnsupportedOperationException(); 210 } 211 212 //???SEC isConnected()213 public synchronized boolean isConnected() { 214 return isConnected; 215 } 216 217 //???SEC isClosed()218 public boolean isClosed() { 219 throw new UnsupportedOperationException(); 220 } 221 222 //???SEC isBound()223 public synchronized boolean isBound() { 224 return isBound; 225 } 226 227 //???SEC isOutputShutdown()228 public boolean isOutputShutdown() { 229 throw new UnsupportedOperationException(); 230 } 231 232 //???SEC isInputShutdown()233 public boolean isInputShutdown() { 234 throw new UnsupportedOperationException(); 235 } 236 237 //???SEC connect(LocalSocketAddress endpoint, int timeout)238 public void connect(LocalSocketAddress endpoint, int timeout) 239 throws IOException { 240 throw new UnsupportedOperationException(); 241 } 242 243 /** 244 * Enqueues a set of file descriptors to send to the peer. The queue 245 * is one deep. The file descriptors will be sent with the next write 246 * of normal data, and will be delivered in a single ancillary message. 247 * See "man 7 unix" SCM_RIGHTS on a desktop Linux machine. 248 * 249 * @param fds non-null; file descriptors to send. 250 */ setFileDescriptorsForSend(FileDescriptor[] fds)251 public void setFileDescriptorsForSend(FileDescriptor[] fds) { 252 impl.setFileDescriptorsForSend(fds); 253 } 254 255 /** 256 * Retrieves a set of file descriptors that a peer has sent through 257 * an ancillary message. This method retrieves the most recent set sent, 258 * and then returns null until a new set arrives. 259 * File descriptors may only be passed along with regular data, so this 260 * method can only return a non-null after a read operation. 261 * 262 * @return null or file descriptor array 263 * @throws IOException 264 */ getAncillaryFileDescriptors()265 public FileDescriptor[] getAncillaryFileDescriptors() throws IOException { 266 return impl.getAncillaryFileDescriptors(); 267 } 268 269 /** 270 * Retrieves the credentials of this socket's peer. Only valid on 271 * connected sockets. 272 * 273 * @return non-null; peer credentials 274 * @throws IOException 275 */ getPeerCredentials()276 public Credentials getPeerCredentials() throws IOException { 277 return impl.getPeerCredentials(); 278 } 279 280 /** 281 * Returns file descriptor or null if not yet open/already closed 282 * 283 * @return fd or null 284 */ getFileDescriptor()285 public FileDescriptor getFileDescriptor() { 286 return impl.getFileDescriptor(); 287 } 288 } 289