1 /* 2 * Copyright (C) 2019 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 com.android.server.am; 18 19 import static android.os.MessageQueue.OnFileDescriptorEventListener.EVENT_ERROR; 20 import static android.os.MessageQueue.OnFileDescriptorEventListener.EVENT_INPUT; 21 22 import static com.android.server.am.ActivityManagerDebugConfig.TAG_AM; 23 import static com.android.server.am.ActivityManagerDebugConfig.TAG_WITH_CLASS_NAME; 24 25 import android.net.LocalSocket; 26 import android.net.LocalSocketAddress; 27 import android.os.MessageQueue; 28 import android.util.Slog; 29 30 import com.android.internal.annotations.GuardedBy; 31 32 import libcore.io.IoUtils; 33 34 import java.io.FileDescriptor; 35 import java.io.IOException; 36 import java.io.InputStream; 37 import java.io.OutputStream; 38 import java.nio.ByteBuffer; 39 40 /** 41 * Lmkd connection to communicate with lowmemorykiller daemon. 42 */ 43 public class LmkdConnection { 44 private static final String TAG = TAG_WITH_CLASS_NAME ? "LmkdConnection" : TAG_AM; 45 46 // lmkd reply max size in bytes 47 private static final int LMKD_REPLY_MAX_SIZE = 12; 48 49 // connection listener interface 50 interface LmkdConnectionListener { onConnect(OutputStream ostream)51 public boolean onConnect(OutputStream ostream); onDisconnect()52 public void onDisconnect(); 53 /** 54 * Check if received reply was expected (reply to an earlier request) 55 * 56 * @param replyBuf The buffer provided in exchange() to receive the reply. 57 * It can be used by exchange() caller to store reply-specific 58 * tags for later use in isReplyExpected() to verify if 59 * received packet is the expected reply. 60 * @param dataReceived The buffer holding received data 61 * @param receivedLen Size of the data received 62 */ isReplyExpected(ByteBuffer replyBuf, ByteBuffer dataReceived, int receivedLen)63 public boolean isReplyExpected(ByteBuffer replyBuf, ByteBuffer dataReceived, 64 int receivedLen); 65 66 /** 67 * Handle the received message if it's unsolicited. 68 * 69 * @param dataReceived The buffer holding received data 70 * @param receivedLen Size of the data received 71 * @return True if the message has been handled correctly, false otherwise. 72 */ handleUnsolicitedMessage(ByteBuffer dataReceived, int receivedLen)73 boolean handleUnsolicitedMessage(ByteBuffer dataReceived, int receivedLen); 74 } 75 76 private final MessageQueue mMsgQueue; 77 78 // lmkd connection listener 79 private final LmkdConnectionListener mListener; 80 81 // mutex to synchronize access to the socket 82 private final Object mLmkdSocketLock = new Object(); 83 84 // socket to communicate with lmkd 85 @GuardedBy("mLmkdSocketLock") 86 private LocalSocket mLmkdSocket = null; 87 88 // socket I/O streams 89 @GuardedBy("mLmkdSocketLock") 90 private OutputStream mLmkdOutputStream = null; 91 @GuardedBy("mLmkdSocketLock") 92 private InputStream mLmkdInputStream = null; 93 94 // buffer to store incoming data 95 private final ByteBuffer mInputBuf = 96 ByteBuffer.allocate(LMKD_REPLY_MAX_SIZE); 97 98 // object to protect mReplyBuf and to wait/notify when reply is received 99 private final Object mReplyBufLock = new Object(); 100 101 // reply buffer 102 @GuardedBy("mReplyBufLock") 103 private ByteBuffer mReplyBuf = null; 104 105 //////////////////// END FIELDS //////////////////// 106 LmkdConnection(MessageQueue msgQueue, LmkdConnectionListener listener)107 LmkdConnection(MessageQueue msgQueue, LmkdConnectionListener listener) { 108 mMsgQueue = msgQueue; 109 mListener = listener; 110 } 111 connect()112 public boolean connect() { 113 synchronized (mLmkdSocketLock) { 114 if (mLmkdSocket != null) { 115 return true; 116 } 117 // temporary sockets and I/O streams 118 final LocalSocket socket = openSocket(); 119 120 if (socket == null) { 121 Slog.w(TAG, "Failed to connect to lowmemorykiller, retry later"); 122 return false; 123 } 124 125 final OutputStream ostream; 126 final InputStream istream; 127 try { 128 ostream = socket.getOutputStream(); 129 istream = socket.getInputStream(); 130 } catch (IOException ex) { 131 IoUtils.closeQuietly(socket); 132 return false; 133 } 134 // execute onConnect callback 135 if (mListener != null && !mListener.onConnect(ostream)) { 136 Slog.w(TAG, "Failed to communicate with lowmemorykiller, retry later"); 137 IoUtils.closeQuietly(socket); 138 return false; 139 } 140 // connection established 141 mLmkdSocket = socket; 142 mLmkdOutputStream = ostream; 143 mLmkdInputStream = istream; 144 mMsgQueue.addOnFileDescriptorEventListener(mLmkdSocket.getFileDescriptor(), 145 EVENT_INPUT | EVENT_ERROR, 146 new MessageQueue.OnFileDescriptorEventListener() { 147 public int onFileDescriptorEvents(FileDescriptor fd, int events) { 148 return fileDescriptorEventHandler(fd, events); 149 } 150 } 151 ); 152 mLmkdSocketLock.notifyAll(); 153 } 154 return true; 155 } 156 fileDescriptorEventHandler(FileDescriptor fd, int events)157 private int fileDescriptorEventHandler(FileDescriptor fd, int events) { 158 if (mListener == null) { 159 return 0; 160 } 161 if ((events & EVENT_INPUT) != 0) { 162 processIncomingData(); 163 } 164 if ((events & EVENT_ERROR) != 0) { 165 synchronized (mLmkdSocketLock) { 166 // stop listening on this socket 167 mMsgQueue.removeOnFileDescriptorEventListener( 168 mLmkdSocket.getFileDescriptor()); 169 IoUtils.closeQuietly(mLmkdSocket); 170 mLmkdSocket = null; 171 } 172 // wake up reply waiters if any 173 synchronized (mReplyBufLock) { 174 if (mReplyBuf != null) { 175 mReplyBuf = null; 176 mReplyBufLock.notifyAll(); 177 } 178 } 179 // notify listener 180 mListener.onDisconnect(); 181 return 0; 182 } 183 return (EVENT_INPUT | EVENT_ERROR); 184 } 185 processIncomingData()186 private void processIncomingData() { 187 int len = read(mInputBuf); 188 if (len > 0) { 189 synchronized (mReplyBufLock) { 190 if (mReplyBuf != null) { 191 if (mListener.isReplyExpected(mReplyBuf, mInputBuf, len)) { 192 // copy into reply buffer 193 mReplyBuf.put(mInputBuf.array(), 0, len); 194 mReplyBuf.rewind(); 195 // wakeup the waiting thread 196 mReplyBufLock.notifyAll(); 197 } else if (!mListener.handleUnsolicitedMessage(mInputBuf, len)) { 198 // received unexpected packet 199 // treat this as an error 200 mReplyBuf = null; 201 mReplyBufLock.notifyAll(); 202 Slog.e(TAG, "Received an unexpected packet from lmkd"); 203 } 204 } else if (!mListener.handleUnsolicitedMessage(mInputBuf, len)) { 205 // received asynchronous communication from lmkd 206 // but we don't recognize it. 207 Slog.w(TAG, "Received an unexpected packet from lmkd"); 208 } 209 } 210 } 211 } 212 isConnected()213 public boolean isConnected() { 214 synchronized (mLmkdSocketLock) { 215 return (mLmkdSocket != null); 216 } 217 } 218 waitForConnection(long timeoutMs)219 public boolean waitForConnection(long timeoutMs) { 220 synchronized (mLmkdSocketLock) { 221 if (mLmkdSocket != null) { 222 return true; 223 } 224 try { 225 mLmkdSocketLock.wait(timeoutMs); 226 return (mLmkdSocket != null); 227 } catch (InterruptedException e) { 228 return false; 229 } 230 } 231 } 232 openSocket()233 private LocalSocket openSocket() { 234 final LocalSocket socket; 235 236 try { 237 socket = new LocalSocket(LocalSocket.SOCKET_SEQPACKET); 238 socket.connect( 239 new LocalSocketAddress("lmkd", 240 LocalSocketAddress.Namespace.RESERVED)); 241 } catch (IOException ex) { 242 Slog.e(TAG, "Connection failed: " + ex.toString()); 243 return null; 244 } 245 return socket; 246 } 247 write(ByteBuffer buf)248 private boolean write(ByteBuffer buf) { 249 synchronized (mLmkdSocketLock) { 250 try { 251 mLmkdOutputStream.write(buf.array(), 0, buf.position()); 252 } catch (IOException ex) { 253 return false; 254 } 255 return true; 256 } 257 } 258 read(ByteBuffer buf)259 private int read(ByteBuffer buf) { 260 synchronized (mLmkdSocketLock) { 261 try { 262 return mLmkdInputStream.read(buf.array(), 0, buf.array().length); 263 } catch (IOException ex) { 264 } 265 return -1; 266 } 267 } 268 269 /** 270 * Exchange a request/reply packets with lmkd 271 * 272 * @param req The buffer holding the request data to be sent 273 * @param repl The buffer to receive the reply 274 */ exchange(ByteBuffer req, ByteBuffer repl)275 public boolean exchange(ByteBuffer req, ByteBuffer repl) { 276 if (repl == null) { 277 return write(req); 278 } 279 280 boolean result = false; 281 // set reply buffer to user-defined one to fill it 282 synchronized (mReplyBufLock) { 283 mReplyBuf = repl; 284 285 if (write(req)) { 286 try { 287 // wait for the reply 288 mReplyBufLock.wait(); 289 result = (mReplyBuf != null); 290 } catch (InterruptedException ie) { 291 result = false; 292 } 293 } 294 295 // reset reply buffer 296 mReplyBuf = null; 297 } 298 return result; 299 } 300 } 301