1 /* 2 * Copyright 2013 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 package com.android.ex.camera2.blocking; 17 18 import android.hardware.camera2.CameraAccessException; 19 import android.hardware.camera2.CameraDevice; 20 import android.hardware.camera2.CameraManager; 21 import android.os.ConditionVariable; 22 import android.os.Handler; 23 import android.os.Looper; 24 import android.util.Log; 25 26 import com.android.ex.camera2.exceptions.TimeoutRuntimeException; 27 28 import java.util.Objects; 29 30 /** 31 * Expose {@link CameraManager} functionality with blocking functions. 32 * 33 * <p>Safe to use at the same time as the regular CameraManager, so this does not 34 * duplicate any functionality that is already blocking.</p> 35 * 36 * <p>Be careful when using this from UI thread! This function will typically block 37 * for about 500ms when successful, and as long as {@value #OPEN_TIME_OUT_MS}ms when timing out.</p> 38 */ 39 public class BlockingCameraManager { 40 41 private static final String TAG = "BlockingCameraManager"; 42 private static final boolean VERBOSE = Log.isLoggable(TAG, Log.VERBOSE); 43 44 private static final int OPEN_TIME_OUT_MS = 2000; // ms time out for openCamera 45 46 /** 47 * Exception thrown by {@link #openCamera} if the open fails asynchronously. 48 */ 49 public static class BlockingOpenException extends Exception { 50 /** 51 * Suppress Eclipse warning 52 */ 53 private static final long serialVersionUID = 12397123891238912L; 54 55 public static final int ERROR_DISCONNECTED = 0; // Does not clash with ERROR_... 56 57 private final int mError; 58 wasDisconnected()59 public boolean wasDisconnected() { 60 return mError == ERROR_DISCONNECTED; 61 } 62 wasError()63 public boolean wasError() { 64 return mError != ERROR_DISCONNECTED; 65 } 66 67 /** 68 * Returns the error code {@link ERROR_DISCONNECTED} if disconnected, or one of 69 * {@code CameraDevice.StateCallback#ERROR_*} if there was another error. 70 * 71 * @return int Disconnect/error code 72 */ getCode()73 public int getCode() { 74 return mError; 75 } 76 77 /** 78 * Thrown when camera device enters error state during open, or if 79 * it disconnects. 80 * 81 * @param errorCode 82 * @param message 83 * 84 * @see {@link CameraDevice.StateCallback#ERROR_CAMERA_DEVICE} 85 */ BlockingOpenException(int errorCode, String message)86 public BlockingOpenException(int errorCode, String message) { 87 super(message); 88 mError = errorCode; 89 } 90 } 91 92 private final CameraManager mManager; 93 94 /** 95 * Create a new blocking camera manager. 96 * 97 * @param manager 98 * CameraManager returned by 99 * {@code Context.getSystemService(Context.CAMERA_SERVICE)} 100 */ BlockingCameraManager(CameraManager manager)101 public BlockingCameraManager(CameraManager manager) { 102 if (manager == null) { 103 throw new IllegalArgumentException("manager must not be null"); 104 } 105 mManager = manager; 106 } 107 108 /** 109 * Open the camera, blocking it until it succeeds or fails. 110 * 111 * <p>Note that the Handler provided must not be null. Furthermore, if there is a handler, 112 * its Looper must not be the current thread's Looper. Otherwise we'd never receive 113 * the callbacks from the CameraDevice since this function would prevent them from being 114 * processed.</p> 115 * 116 * <p>Throws {@link CameraAccessException} for the same reason {@link CameraManager#openCamera} 117 * does.</p> 118 * 119 * <p>Throws {@link BlockingOpenException} when the open fails asynchronously (due to 120 * {@link CameraDevice.StateCallback#onDisconnected(CameraDevice)} or 121 * ({@link CameraDevice.StateCallback#onError(CameraDevice)}.</p> 122 * 123 * <p>Throws {@link TimeoutRuntimeException} if opening times out. This is usually 124 * highly unrecoverable, and all future calls to opening that camera will fail since the 125 * service will think it's busy. This class will do its best to clean up eventually.</p> 126 * 127 * @param cameraId 128 * Id of the camera 129 * @param listener 130 * Listener to the camera. onOpened, onDisconnected, onError need not be implemented. 131 * @param handler 132 * Handler which to run the listener on. Must not be null. 133 * 134 * @return CameraDevice 135 * 136 * @throws IllegalArgumentException 137 * If the handler is null, or if the handler's looper is current. 138 * @throws CameraAccessException 139 * If open fails immediately. 140 * @throws BlockingOpenException 141 * If open fails after blocking for some amount of time. 142 * @throws TimeoutRuntimeException 143 * If opening times out. Typically unrecoverable. 144 */ openCamera(String cameraId, CameraDevice.StateCallback listener, Handler handler)145 public CameraDevice openCamera(String cameraId, CameraDevice.StateCallback listener, 146 Handler handler) throws CameraAccessException, BlockingOpenException { 147 148 if (handler == null) { 149 throw new IllegalArgumentException("handler must not be null"); 150 } else if (handler.getLooper() == Looper.myLooper()) { 151 throw new IllegalArgumentException("handler's looper must not be the current looper"); 152 } 153 154 return (new OpenListener(mManager, cameraId, listener, handler)).blockUntilOpen(); 155 } 156 assertEquals(Object a, Object b)157 private static void assertEquals(Object a, Object b) { 158 if (!Objects.equals(a, b)) { 159 throw new AssertionError("Expected " + a + ", but got " + b); 160 } 161 } 162 163 /** 164 * Block until CameraManager#openCamera finishes with onOpened/onError/onDisconnected 165 * 166 * <p>Pass-through all StateCallback changes to the proxy.</p> 167 * 168 * <p>Time out after {@link #OPEN_TIME_OUT_MS} and unblock. Clean up camera if it arrives 169 * later.</p> 170 */ 171 private class OpenListener extends CameraDevice.StateCallback { 172 private static final int ERROR_UNINITIALIZED = -1; 173 174 private final String mCameraId; 175 176 private final CameraDevice.StateCallback mProxy; 177 178 private final Object mLock = new Object(); 179 private final ConditionVariable mDeviceReady = new ConditionVariable(); 180 181 private CameraDevice mDevice = null; 182 private boolean mSuccess = false; 183 private int mError = ERROR_UNINITIALIZED; 184 private boolean mDisconnected = false; 185 186 private boolean mNoReply = true; // Start with no reply until proven otherwise 187 private boolean mTimedOut = false; 188 OpenListener(CameraManager manager, String cameraId, CameraDevice.StateCallback listener, Handler handler)189 OpenListener(CameraManager manager, String cameraId, 190 CameraDevice.StateCallback listener, Handler handler) 191 throws CameraAccessException { 192 mCameraId = cameraId; 193 mProxy = listener; 194 manager.openCamera(cameraId, this, handler); 195 } 196 197 // Freebie check to make sure we aren't calling functions multiple times. 198 // We should still test the state interactions in a separate more thorough test. assertInitialState()199 private void assertInitialState() { 200 assertEquals(null, mDevice); 201 assertEquals(false, mDisconnected); 202 assertEquals(ERROR_UNINITIALIZED, mError); 203 assertEquals(false, mSuccess); 204 } 205 206 @Override onOpened(CameraDevice camera)207 public void onOpened(CameraDevice camera) { 208 if (VERBOSE) { 209 Log.v(TAG, "onOpened: camera " + ((camera != null) ? camera.getId() : "null")); 210 } 211 212 synchronized (mLock) { 213 assertInitialState(); 214 mNoReply = false; 215 mSuccess = true; 216 mDevice = camera; 217 mDeviceReady.open(); 218 219 if (mTimedOut && camera != null) { 220 camera.close(); 221 return; 222 } 223 } 224 225 if (mProxy != null) mProxy.onOpened(camera); 226 } 227 228 @Override onDisconnected(CameraDevice camera)229 public void onDisconnected(CameraDevice camera) { 230 if (VERBOSE) { 231 Log.v(TAG, "onDisconnected: camera " 232 + ((camera != null) ? camera.getId() : "null")); 233 } 234 235 synchronized (mLock) { 236 assertInitialState(); 237 mNoReply = false; 238 mDisconnected = true; 239 mDevice = camera; 240 mDeviceReady.open(); 241 242 if (mTimedOut && camera != null) { 243 camera.close(); 244 return; 245 } 246 } 247 248 if (mProxy != null) mProxy.onDisconnected(camera); 249 } 250 251 @Override onError(CameraDevice camera, int error)252 public void onError(CameraDevice camera, int error) { 253 if (VERBOSE) { 254 Log.v(TAG, "onError: camera " + ((camera != null) ? camera.getId() : "null")); 255 } 256 257 if (error <= 0) { 258 throw new AssertionError("Expected error to be a positive number"); 259 } 260 261 synchronized (mLock) { 262 // Don't assert initial state. Error can happen later. 263 mNoReply = false; 264 mError = error; 265 mDevice = camera; 266 mDeviceReady.open(); 267 268 if (mTimedOut && camera != null) { 269 camera.close(); 270 return; 271 } 272 } 273 274 if (mProxy != null) mProxy.onError(camera, error); 275 } 276 277 @Override onClosed(CameraDevice camera)278 public void onClosed(CameraDevice camera) { 279 if (mProxy != null) mProxy.onClosed(camera); 280 } 281 blockUntilOpen()282 CameraDevice blockUntilOpen() throws BlockingOpenException { 283 /** 284 * Block until onOpened, onError, or onDisconnected 285 */ 286 if (!mDeviceReady.block(OPEN_TIME_OUT_MS)) { 287 288 synchronized (mLock) { 289 if (mNoReply) { // Give the async camera a fighting chance (required) 290 mTimedOut = true; // Clean up camera if it ever arrives later 291 throw new TimeoutRuntimeException(String.format( 292 "Timed out after %d ms while trying to open camera device %s", 293 OPEN_TIME_OUT_MS, mCameraId)); 294 } 295 } 296 } 297 298 synchronized (mLock) { 299 /** 300 * Determine which state we ended up in: 301 * 302 * - Throw exceptions for onError/onDisconnected 303 * - Return device for onOpened 304 */ 305 if (!mSuccess && mDevice != null) { 306 mDevice.close(); 307 } 308 309 if (mSuccess) { 310 return mDevice; 311 } else { 312 if (mDisconnected) { 313 throw new BlockingOpenException( 314 BlockingOpenException.ERROR_DISCONNECTED, 315 "Failed to open camera device: it is disconnected"); 316 } else if (mError != ERROR_UNINITIALIZED) { 317 throw new BlockingOpenException( 318 mError, 319 "Failed to open camera device: error code " + mError); 320 } else { 321 throw new AssertionError("Failed to open camera device (impl bug)"); 322 } 323 } 324 } 325 } 326 } 327 } 328