blob: 65f950228723fd60261c53ce80d2907e151d03a1 [file] [log] [blame]
/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License
*/
package com.android.server.backup.transport;
import android.annotation.IntDef;
import android.annotation.Nullable;
import android.annotation.WorkerThread;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.DeadObjectException;
import android.os.Handler;
import android.os.IBinder;
import android.os.Looper;
import android.os.UserHandle;
import android.util.ArrayMap;
import android.util.Log;
import com.android.internal.annotations.GuardedBy;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.backup.IBackupTransport;
import com.android.internal.util.Preconditions;
import com.android.server.backup.TransportManager;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
/**
* A {@link TransportClient} manages the connection to an {@link IBackupTransport} service, obtained
* via the {@param bindIntent} parameter provided in the constructor. A {@link TransportClient} is
* responsible for only one connection to the transport service, not more.
*
* <p>After retrieved using {@link TransportManager#getTransportClient(String, String)}, you can
* call either {@link #connect(String)}, if you can block your thread, or {@link
* #connectAsync(TransportConnectionListener, String)}, otherwise, to obtain a {@link
* IBackupTransport} instance. It's meant to be passed around as a token to a connected transport.
* When the connection is not needed anymore you should call {@link #unbind(String)} or indirectly
* via {@link TransportManager#disposeOfTransportClient(TransportClient, String)}.
*
* <p>DO NOT forget to unbind otherwise there will be dangling connections floating around.
*
* <p>This class is thread-safe.
*
* @see TransportManager
*/
public class TransportClient {
private static final String TAG = "TransportClient";
private final Context mContext;
private final Intent mBindIntent;
private final String mIdentifier;
private final ComponentName mTransportComponent;
private final Handler mListenerHandler;
private final String mPrefixForLog;
private final Object mStateLock = new Object();
@GuardedBy("mStateLock")
private final Map<TransportConnectionListener, String> mListeners = new ArrayMap<>();
@GuardedBy("mStateLock")
@State
private int mState = State.IDLE;
@GuardedBy("mStateLock")
private volatile IBackupTransport mTransport;
TransportClient(
Context context,
Intent bindIntent,
ComponentName transportComponent,
String identifier) {
this(context, bindIntent, transportComponent, identifier, Handler.getMain());
}
@VisibleForTesting
TransportClient(
Context context,
Intent bindIntent,
ComponentName transportComponent,
String identifier,
Handler listenerHandler) {
mContext = context;
mTransportComponent = transportComponent;
mBindIntent = bindIntent;
mIdentifier = identifier;
mListenerHandler = listenerHandler;
// For logging
String classNameForLog = mTransportComponent.getShortClassName().replaceFirst(".*\\.", "");
mPrefixForLog = classNameForLog + "#" + mIdentifier + ": ";
}
public ComponentName getTransportComponent() {
return mTransportComponent;
}
// Calls to onServiceDisconnected() or onBindingDied() turn TransportClient UNUSABLE. After one
// of these calls, if a binding happen again the new service can be a different instance. Since
// transports are stateful, we don't want a new instance responding for an old instance's state.
private ServiceConnection mConnection =
new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName componentName, IBinder binder) {
IBackupTransport transport = IBackupTransport.Stub.asInterface(binder);
synchronized (mStateLock) {
checkStateIntegrityLocked();
if (mState != State.UNUSABLE) {
log(Log.DEBUG, "Transport connected");
setStateLocked(State.CONNECTED, transport);
notifyListenersAndClearLocked(transport);
}
}
}
@Override
public void onServiceDisconnected(ComponentName componentName) {
synchronized (mStateLock) {
log(Log.ERROR, "Service disconnected: client UNUSABLE");
setStateLocked(State.UNUSABLE, null);
// After unbindService() no calls back to mConnection
mContext.unbindService(this);
}
}
@Override
public void onBindingDied(ComponentName name) {
synchronized (mStateLock) {
checkStateIntegrityLocked();
log(Log.ERROR, "Binding died: client UNUSABLE");
// After unbindService() no calls back to mConnection
switch (mState) {
case State.UNUSABLE:
break;
case State.IDLE:
log(Log.ERROR, "Unexpected state transition IDLE => UNUSABLE");
setStateLocked(State.UNUSABLE, null);
break;
case State.BOUND_AND_CONNECTING:
setStateLocked(State.UNUSABLE, null);
mContext.unbindService(this);
notifyListenersAndClearLocked(null);
break;
case State.CONNECTED:
setStateLocked(State.UNUSABLE, null);
mContext.unbindService(this);
break;
}
}
}
};
/**
* Attempts to connect to the transport (if needed).
*
* <p>Note that being bound is not the same as connected. To be connected you also need to be
* bound. You go from nothing to bound, then to bound and connected. To have a usable transport
* binder instance you need to be connected. This method will attempt to connect and return an
* usable transport binder regardless of the state of the object, it may already be connected,
* or bound but not connected, not bound at all or even unusable.
*
* <p>So, a {@link Context#bindServiceAsUser(Intent, ServiceConnection, int, UserHandle)} (or
* one of its variants) can be called or not depending on the inner state. However, it won't be
* called again if we're already bound. For example, if one was already requested but the
* framework has not yet returned (meaning we're bound but still trying to connect) it won't
* trigger another one, just piggyback on the original request.
*
* <p>It's guaranteed that you are going to get a call back to {@param listener} after this
* call. However, the {@param IBackupTransport} parameter, the transport binder, is not
* guaranteed to be non-null, or if it's non-null it's not guaranteed to be usable - i.e. it can
* throw {@link DeadObjectException}s on method calls. You should check for both in your code.
* The reasons for a null transport binder are:
*
* <ul>
* <li>Some code called {@link #unbind(String)} before you got a callback.
* <li>The framework had already called {@link
* ServiceConnection#onServiceDisconnected(ComponentName)} or {@link
* ServiceConnection#onBindingDied(ComponentName)} on this object's connection before.
* Check the documentation of those methods for when that happens.
* <li>The framework returns false for {@link Context#bindServiceAsUser(Intent,
* ServiceConnection, int, UserHandle)} (or one of its variants). Check documentation for
* when this happens.
* </ul>
*
* For unusable transport binders check {@link DeadObjectException}.
*
* @param listener The listener that will be called with the (possibly null or unusable) {@link
* IBackupTransport} instance and this {@link TransportClient} object.
* @param caller A {@link String} identifying the caller for logging/debugging purposes. This
* should be a human-readable short string that is easily identifiable in the logs. Ideally
* TAG.methodName(), where TAG is the one used in logcat. In cases where this is is not very
* descriptive like MyHandler.handleMessage() you should put something that someone reading
* the code would understand, like MyHandler/MSG_FOO.
* @see #connect(String)
* @see DeadObjectException
* @see ServiceConnection#onServiceConnected(ComponentName, IBinder)
* @see ServiceConnection#onServiceDisconnected(ComponentName)
* @see Context#bindServiceAsUser(Intent, ServiceConnection, int, UserHandle)
*/
public void connectAsync(TransportConnectionListener listener, String caller) {
synchronized (mStateLock) {
checkStateIntegrityLocked();
switch (mState) {
case State.UNUSABLE:
log(Log.DEBUG, caller, "Async connect: UNUSABLE client");
notifyListener(listener, null, caller);
break;
case State.IDLE:
boolean hasBound =
mContext.bindServiceAsUser(
mBindIntent,
mConnection,
Context.BIND_AUTO_CREATE,
TransportManager.createSystemUserHandle());
if (hasBound) {
// We don't need to set a time-out because we are guaranteed to get a call
// back in ServiceConnection, either an onServiceConnected() or
// onBindingDied().
log(Log.DEBUG, caller, "Async connect: service bound, connecting");
setStateLocked(State.BOUND_AND_CONNECTING, null);
mListeners.put(listener, caller);
} else {
log(Log.ERROR, "Async connect: bindService returned false");
// mState remains State.IDLE
mContext.unbindService(mConnection);
notifyListener(listener, null, caller);
}
break;
case State.BOUND_AND_CONNECTING:
log(Log.DEBUG, caller, "Async connect: already connecting, adding listener");
mListeners.put(listener, caller);
break;
case State.CONNECTED:
log(Log.DEBUG, caller, "Async connect: reusing transport");
notifyListener(listener, mTransport, caller);
break;
}
}
}
/**
* Removes the transport binding.
*
* @param caller A {@link String} identifying the caller for logging/debugging purposes. Check
* {@link #connectAsync(TransportConnectionListener, String)} for more details.
*/
public void unbind(String caller) {
synchronized (mStateLock) {
checkStateIntegrityLocked();
log(Log.DEBUG, caller, "Unbind requested (was " + stateToString(mState) + ")");
switch (mState) {
case State.UNUSABLE:
case State.IDLE:
break;
case State.BOUND_AND_CONNECTING:
setStateLocked(State.IDLE, null);
// After unbindService() no calls back to mConnection
mContext.unbindService(mConnection);
notifyListenersAndClearLocked(null);
break;
case State.CONNECTED:
setStateLocked(State.IDLE, null);
mContext.unbindService(mConnection);
break;
}
}
}
/**
* Attempts to connect to the transport (if needed) and returns it.
*
* <p>Synchronous version of {@link #connectAsync(TransportConnectionListener, String)}. The
* same observations about state are valid here. Also, what was said about the {@link
* IBackupTransport} parameter of {@link TransportConnectionListener} now apply to the return
* value of this method.
*
* <p>This is a potentially blocking operation, so be sure to call this carefully on the correct
* threads. You can't call this from the process main-thread (it throws an exception if you do
* so).
*
* <p>In most cases only the first call to this method will block, the following calls should
* return instantly. However, this is not guaranteed.
*
* @param caller A {@link String} identifying the caller for logging/debugging purposes. Check
* {@link #connectAsync(TransportConnectionListener, String)} for more details.
* @return A {@link IBackupTransport} transport binder instance or null. If it's non-null it can
* still be unusable - throws {@link DeadObjectException} on method calls
*/
@WorkerThread
@Nullable
public IBackupTransport connect(String caller) {
// If called on the main-thread this could deadlock waiting because calls to
// ServiceConnection are on the main-thread as well
Preconditions.checkState(
!Looper.getMainLooper().isCurrentThread(), "Can't call connect() on main thread");
IBackupTransport transport = mTransport;
if (transport != null) {
log(Log.DEBUG, caller, "Sync connect: reusing transport");
return transport;
}
// If it's already UNUSABLE we return straight away, no need to go to main-thread
synchronized (mStateLock) {
if (mState == State.UNUSABLE) {
log(Log.DEBUG, caller, "Sync connect: UNUSABLE client");
return null;
}
}
CompletableFuture<IBackupTransport> transportFuture = new CompletableFuture<>();
TransportConnectionListener requestListener =
(requestedTransport, transportClient) ->
transportFuture.complete(requestedTransport);
log(Log.DEBUG, caller, "Sync connect: calling async");
connectAsync(requestListener, caller);
try {
return transportFuture.get();
} catch (InterruptedException | ExecutionException e) {
String error = e.getClass().getSimpleName();
log(Log.ERROR, caller, error + " while waiting for transport: " + e.getMessage());
return null;
}
}
/**
* Tries to connect to the transport, if it fails throws {@link TransportNotAvailableException}.
*
* <p>Same as {@link #connect(String)} except it throws instead of returning null.
*
* @param caller A {@link String} identifying the caller for logging/debugging purposes. Check
* {@link #connectAsync(TransportConnectionListener, String)} for more details.
* @return A {@link IBackupTransport} transport binder instance.
* @see #connect(String)
* @throws TransportNotAvailableException if connection attempt fails.
*/
@WorkerThread
public IBackupTransport connectOrThrow(String caller) throws TransportNotAvailableException {
IBackupTransport transport = connect(caller);
if (transport == null) {
log(Log.ERROR, caller, "Transport connection failed");
throw new TransportNotAvailableException();
}
return transport;
}
/**
* If the {@link TransportClient} is already connected to the transport, returns the transport,
* otherwise throws {@link TransportNotAvailableException}.
*
* @param caller A {@link String} identifying the caller for logging/debugging purposes. Check
* {@link #connectAsync(TransportConnectionListener, String)} for more details.
* @return A {@link IBackupTransport} transport binder instance.
* @throws TransportNotAvailableException if not connected.
*/
public IBackupTransport getConnectedTransport(String caller)
throws TransportNotAvailableException {
IBackupTransport transport = mTransport;
if (transport == null) {
log(Log.ERROR, caller, "Transport not connected");
throw new TransportNotAvailableException();
}
return transport;
}
@Override
public String toString() {
return "TransportClient{"
+ mTransportComponent.flattenToShortString()
+ "#"
+ mIdentifier
+ "}";
}
private void notifyListener(
TransportConnectionListener listener, IBackupTransport transport, String caller) {
log(Log.VERBOSE, caller, "Notifying listener of transport = " + transport);
mListenerHandler.post(() -> listener.onTransportConnectionResult(transport, this));
}
@GuardedBy("mStateLock")
private void notifyListenersAndClearLocked(IBackupTransport transport) {
for (Map.Entry<TransportConnectionListener, String> entry : mListeners.entrySet()) {
TransportConnectionListener listener = entry.getKey();
String caller = entry.getValue();
notifyListener(listener, transport, caller);
}
mListeners.clear();
}
@GuardedBy("mStateLock")
private void setStateLocked(@State int state, @Nullable IBackupTransport transport) {
log(Log.VERBOSE, "State: " + stateToString(mState) + " => " + stateToString(state));
mState = state;
mTransport = transport;
}
@GuardedBy("mStateLock")
private void checkStateIntegrityLocked() {
switch (mState) {
case State.UNUSABLE:
checkState(mListeners.isEmpty(), "Unexpected listeners when state = UNUSABLE");
checkState(
mTransport == null, "Transport expected to be null when state = UNUSABLE");
case State.IDLE:
checkState(mListeners.isEmpty(), "Unexpected listeners when state = IDLE");
checkState(mTransport == null, "Transport expected to be null when state = IDLE");
break;
case State.BOUND_AND_CONNECTING:
checkState(
mTransport == null,
"Transport expected to be null when state = BOUND_AND_CONNECTING");
break;
case State.CONNECTED:
checkState(mListeners.isEmpty(), "Unexpected listeners when state = CONNECTED");
checkState(
mTransport != null,
"Transport expected to be non-null when state = CONNECTED");
break;
default:
checkState(false, "Unexpected state = " + stateToString(mState));
}
}
private void checkState(boolean assertion, String message) {
if (!assertion) {
log(Log.ERROR, message);
}
}
private String stateToString(@State int state) {
switch (state) {
case State.UNUSABLE:
return "UNUSABLE";
case State.IDLE:
return "IDLE";
case State.BOUND_AND_CONNECTING:
return "BOUND_AND_CONNECTING";
case State.CONNECTED:
return "CONNECTED";
default:
return "<UNKNOWN = " + state + ">";
}
}
private void log(int priority, String message) {
TransportUtils.log(priority, TAG, message);
}
private void log(int priority, String caller, String msg) {
TransportUtils.log(priority, TAG, mPrefixForLog, caller, msg);
// TODO(brufino): Log in internal list for dump
// CharSequence time = DateFormat.format("yyyy-MM-dd HH:mm:ss", System.currentTimeMillis());
}
@IntDef({State.UNUSABLE, State.IDLE, State.BOUND_AND_CONNECTING, State.CONNECTED})
@Retention(RetentionPolicy.SOURCE)
private @interface State {
int UNUSABLE = 0;
int IDLE = 1;
int BOUND_AND_CONNECTING = 2;
int CONNECTED = 3;
}
}