1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not 5 * use this file except in compliance with the License. You may obtain a copy of 6 * 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, WITHOUT 12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 13 * License for the specific language governing permissions and limitations under 14 * the License. 15 */ 16 package android.os; 17 18 import android.annotation.NonNull; 19 import android.annotation.Nullable; 20 21 import java.util.concurrent.CompletableFuture; 22 import java.util.concurrent.ExecutionException; 23 import java.util.concurrent.TimeUnit; 24 import java.util.concurrent.TimeoutException; 25 26 /** 27 * Extends ResultReceiver to allow the server end of the ResultReceiver to synchronously wait 28 * on the response from the client. This enables an RPC like system but with the ability to 29 * timeout and discard late results. 30 * 31 * NOTE: Can only be used for one response. Subsequent responses on the same instance are ignored. 32 * {@hide} 33 */ 34 public class SynchronousResultReceiver extends ResultReceiver { 35 public static class Result { 36 public int resultCode; 37 @Nullable public Bundle bundle; 38 39 public Result(int resultCode, @Nullable Bundle bundle) { 40 this.resultCode = resultCode; 41 this.bundle = bundle; 42 } 43 } 44 45 private final CompletableFuture<Result> mFuture = new CompletableFuture<>(); 46 private final String mName; 47 48 public SynchronousResultReceiver() { 49 super((Handler) null); 50 mName = null; 51 } 52 53 /** 54 * @param name Name for logging purposes 55 */ 56 public SynchronousResultReceiver(String name) { 57 super((Handler) null); 58 mName = name; 59 } 60 61 @Override 62 final protected void onReceiveResult(int resultCode, Bundle resultData) { 63 super.onReceiveResult(resultCode, resultData); 64 mFuture.complete(new Result(resultCode, resultData)); 65 } 66 67 public String getName() { 68 return mName; 69 } 70 71 /** 72 * Blocks waiting for the result from the remote client. 73 * 74 * @return the Result 75 * @throws TimeoutException if the timeout in milliseconds expired. 76 */ 77 public @NonNull Result awaitResult(long timeoutMillis) throws TimeoutException { 78 final long deadline = System.currentTimeMillis() + timeoutMillis; 79 while (timeoutMillis >= 0) { 80 try { 81 return mFuture.get(timeoutMillis, TimeUnit.MILLISECONDS); 82 } catch (ExecutionException e) { 83 // This will NEVER happen. 84 throw new AssertionError("Error receiving response", e); 85 } catch (InterruptedException e) { 86 // The thread was interrupted, try and get the value again, this time 87 // with the remaining time until the deadline. 88 timeoutMillis -= deadline - System.currentTimeMillis(); 89 } 90 } 91 throw new TimeoutException(); 92 } 93 94 } 95