1 /* 2 * Copyright (C) 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 17 package android.hardware.camera2.utils; 18 19 import static android.hardware.camera2.CameraAccessException.CAMERA_DISABLED; 20 import static android.hardware.camera2.CameraAccessException.CAMERA_DISCONNECTED; 21 import static android.hardware.camera2.CameraAccessException.CAMERA_IN_USE; 22 import static android.hardware.camera2.CameraAccessException.CAMERA_ERROR; 23 import static android.hardware.camera2.CameraAccessException.MAX_CAMERAS_IN_USE; 24 import static android.hardware.camera2.CameraAccessException.CAMERA_DEPRECATED_HAL; 25 26 import android.os.DeadObjectException; 27 import android.os.RemoteException; 28 29 import java.lang.reflect.Method; 30 31 /** 32 * Translate camera device status_t return values into exceptions. 33 * 34 * @see android.hardware.camera2.utils.CameraBinderDecorator#newInstance 35 * @hide 36 */ 37 public class CameraBinderDecorator { 38 39 public static final int NO_ERROR = 0; 40 public static final int PERMISSION_DENIED = -1; 41 public static final int ALREADY_EXISTS = -17; 42 public static final int BAD_VALUE = -22; 43 public static final int DEAD_OBJECT = -32; 44 public static final int INVALID_OPERATION = -38; 45 public static final int TIMED_OUT = -110; 46 47 /** 48 * TODO: add as error codes in Errors.h 49 * - POLICY_PROHIBITS 50 * - RESOURCE_BUSY 51 * - NO_SUCH_DEVICE 52 * - NOT_SUPPORTED 53 * - TOO_MANY_USERS 54 */ 55 public static final int EACCES = -13; 56 public static final int EBUSY = -16; 57 public static final int ENODEV = -19; 58 public static final int EOPNOTSUPP = -95; 59 public static final int EUSERS = -87; 60 61 62 static class CameraBinderDecoratorListener implements Decorator.DecoratorListener { 63 64 @Override onBeforeInvocation(Method m, Object[] args)65 public void onBeforeInvocation(Method m, Object[] args) { 66 } 67 68 @Override onAfterInvocation(Method m, Object[] args, Object result)69 public void onAfterInvocation(Method m, Object[] args, Object result) { 70 // int return type => status_t => convert to exception 71 if (m.getReturnType() == Integer.TYPE) { 72 int returnValue = (Integer) result; 73 throwOnError(returnValue); 74 } 75 } 76 77 @Override onCatchException(Method m, Object[] args, Throwable t)78 public boolean onCatchException(Method m, Object[] args, Throwable t) { 79 80 if (t instanceof DeadObjectException) { 81 throw new CameraRuntimeException(CAMERA_DISCONNECTED, 82 "Process hosting the camera service has died unexpectedly", 83 t); 84 } else if (t instanceof RemoteException) { 85 throw new UnsupportedOperationException("An unknown RemoteException was thrown" + 86 " which should never happen.", t); 87 } 88 89 return false; 90 } 91 92 @Override onFinally(Method m, Object[] args)93 public void onFinally(Method m, Object[] args) { 94 } 95 96 } 97 98 /** 99 * Throw error codes returned by the camera service as exceptions. 100 * 101 * @param errorFlag error to throw as an exception. 102 */ throwOnError(int errorFlag)103 public static void throwOnError(int errorFlag) { 104 switch (errorFlag) { 105 case NO_ERROR: 106 return; 107 case PERMISSION_DENIED: 108 throw new SecurityException("Lacking privileges to access camera service"); 109 case ALREADY_EXISTS: 110 // This should be handled at the call site. Typically this isn't bad, 111 // just means we tried to do an operation that already completed. 112 return; 113 case BAD_VALUE: 114 throw new IllegalArgumentException("Bad argument passed to camera service"); 115 case DEAD_OBJECT: 116 throw new CameraRuntimeException(CAMERA_DISCONNECTED); 117 case TIMED_OUT: 118 throw new CameraRuntimeException(CAMERA_ERROR, 119 "Operation timed out in camera service"); 120 case EACCES: 121 throw new CameraRuntimeException(CAMERA_DISABLED); 122 case EBUSY: 123 throw new CameraRuntimeException(CAMERA_IN_USE); 124 case EUSERS: 125 throw new CameraRuntimeException(MAX_CAMERAS_IN_USE); 126 case ENODEV: 127 throw new CameraRuntimeException(CAMERA_DISCONNECTED); 128 case EOPNOTSUPP: 129 throw new CameraRuntimeException(CAMERA_DEPRECATED_HAL); 130 case INVALID_OPERATION: 131 throw new CameraRuntimeException(CAMERA_ERROR, 132 "Illegal state encountered in camera service."); 133 } 134 135 /** 136 * Trap the rest of the negative return values. If we have known 137 * error codes i.e. ALREADY_EXISTS that aren't really runtime 138 * errors, then add them to the top switch statement 139 */ 140 if (errorFlag < 0) { 141 throw new UnsupportedOperationException(String.format("Unknown error %d", 142 errorFlag)); 143 } 144 } 145 146 /** 147 * <p> 148 * Wraps the type T with a proxy that will check 'status_t' return codes 149 * from the native side of the camera service, and throw Java exceptions 150 * automatically based on the code. 151 * </p> 152 * <p> 153 * In addition it also rewrites binder's RemoteException into either a 154 * CameraAccessException or an UnsupportedOperationException. 155 * </p> 156 * <p> 157 * As a result of calling any method on the proxy, RemoteException is 158 * guaranteed never to be thrown. 159 * </p> 160 * 161 * @param obj object that will serve as the target for all method calls 162 * @param <T> the type of the element you want to wrap. This must be an interface. 163 * @return a proxy that will intercept all invocations to obj 164 */ newInstance(T obj)165 public static <T> T newInstance(T obj) { 166 return Decorator.<T> newInstance(obj, new CameraBinderDecoratorListener()); 167 } 168 } 169