1 /* 2 * Copyright (C) 2009 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.camera; 18 19 import static com.android.camera.Util.Assert; 20 21 import android.hardware.Camera.CameraInfo; 22 import android.hardware.Camera.Parameters; 23 import android.os.Build; 24 import android.os.Handler; 25 import android.os.HandlerThread; 26 import android.os.Looper; 27 import android.os.Message; 28 import android.util.Log; 29 30 import com.android.camera.CameraManager.CameraProxy; 31 32 import java.io.IOException; 33 import java.text.SimpleDateFormat; 34 import java.util.ArrayList; 35 import java.util.Date; 36 37 /** 38 * The class is used to hold an {@code android.hardware.Camera} instance. 39 * 40 * <p>The {@code open()} and {@code release()} calls are similar to the ones 41 * in {@code android.hardware.Camera}. The difference is if {@code keep()} is 42 * called before {@code release()}, CameraHolder will try to hold the {@code 43 * android.hardware.Camera} instance for a while, so if {@code open()} is 44 * called soon after, we can avoid the cost of {@code open()} in {@code 45 * android.hardware.Camera}. 46 * 47 * <p>This is used in switching between different modules. 48 */ 49 public class CameraHolder { 50 private static final String TAG = "CameraHolder"; 51 private static final int KEEP_CAMERA_TIMEOUT = 3000; // 3 seconds 52 private CameraProxy mCameraDevice; 53 private long mKeepBeforeTime; // Keep the Camera before this time. 54 private final Handler mHandler; 55 private boolean mCameraOpened; // true if camera is opened 56 private final int mNumberOfCameras; 57 private int mCameraId = -1; // current camera id 58 private int mBackCameraId = -1; 59 private int mFrontCameraId = -1; 60 private final CameraInfo[] mInfo; 61 private static CameraProxy mMockCamera[]; 62 private static CameraInfo mMockCameraInfo[]; 63 64 /* Debug double-open issue */ 65 private static final boolean DEBUG_OPEN_RELEASE = true; 66 private static class OpenReleaseState { 67 long time; 68 int id; 69 String device; 70 String[] stack; 71 } 72 private static ArrayList<OpenReleaseState> sOpenReleaseStates = 73 new ArrayList<OpenReleaseState>(); 74 private static SimpleDateFormat sDateFormat = new SimpleDateFormat( 75 "yyyy-MM-dd HH:mm:ss.SSS"); 76 collectState(int id, CameraProxy device)77 private static synchronized void collectState(int id, CameraProxy device) { 78 OpenReleaseState s = new OpenReleaseState(); 79 s.time = System.currentTimeMillis(); 80 s.id = id; 81 if (device == null) { 82 s.device = "(null)"; 83 } else { 84 s.device = device.toString(); 85 } 86 87 StackTraceElement[] stack = Thread.currentThread().getStackTrace(); 88 String[] lines = new String[stack.length]; 89 for (int i = 0; i < stack.length; i++) { 90 lines[i] = stack[i].toString(); 91 } 92 s.stack = lines; 93 94 if (sOpenReleaseStates.size() > 10) { 95 sOpenReleaseStates.remove(0); 96 } 97 sOpenReleaseStates.add(s); 98 } 99 dumpStates()100 private static synchronized void dumpStates() { 101 for (int i = sOpenReleaseStates.size() - 1; i >= 0; i--) { 102 OpenReleaseState s = sOpenReleaseStates.get(i); 103 String date = sDateFormat.format(new Date(s.time)); 104 Log.d(TAG, "State " + i + " at " + date); 105 Log.d(TAG, "mCameraId = " + s.id + ", mCameraDevice = " + s.device); 106 Log.d(TAG, "Stack:"); 107 for (int j = 0; j < s.stack.length; j++) { 108 Log.d(TAG, " " + s.stack[j]); 109 } 110 } 111 } 112 113 // We store the camera parameters when we actually open the device, 114 // so we can restore them in the subsequent open() requests by the user. 115 // This prevents the parameters set by PhotoModule used by VideoModule 116 // inadvertently. 117 private Parameters mParameters; 118 119 // Use a singleton. 120 private static CameraHolder sHolder; instance()121 public static synchronized CameraHolder instance() { 122 if (sHolder == null) { 123 sHolder = new CameraHolder(); 124 } 125 return sHolder; 126 } 127 128 private static final int RELEASE_CAMERA = 1; 129 private class MyHandler extends Handler { MyHandler(Looper looper)130 MyHandler(Looper looper) { 131 super(looper); 132 } 133 134 @Override handleMessage(Message msg)135 public void handleMessage(Message msg) { 136 switch(msg.what) { 137 case RELEASE_CAMERA: 138 synchronized (CameraHolder.this) { 139 // In 'CameraHolder.open', the 'RELEASE_CAMERA' message 140 // will be removed if it is found in the queue. However, 141 // there is a chance that this message has been handled 142 // before being removed. So, we need to add a check 143 // here: 144 if (!mCameraOpened) release(); 145 } 146 break; 147 } 148 } 149 } 150 injectMockCamera(CameraInfo[] info, CameraProxy[] camera)151 public static void injectMockCamera(CameraInfo[] info, CameraProxy[] camera) { 152 mMockCameraInfo = info; 153 mMockCamera = camera; 154 sHolder = new CameraHolder(); 155 } 156 CameraHolder()157 private CameraHolder() { 158 HandlerThread ht = new HandlerThread("CameraHolder"); 159 ht.start(); 160 mHandler = new MyHandler(ht.getLooper()); 161 if (mMockCameraInfo != null) { 162 mNumberOfCameras = mMockCameraInfo.length; 163 mInfo = mMockCameraInfo; 164 } else { 165 mNumberOfCameras = android.hardware.Camera.getNumberOfCameras(); 166 mInfo = new CameraInfo[mNumberOfCameras]; 167 for (int i = 0; i < mNumberOfCameras; i++) { 168 mInfo[i] = new CameraInfo(); 169 android.hardware.Camera.getCameraInfo(i, mInfo[i]); 170 } 171 } 172 173 // get the first (smallest) back and first front camera id 174 for (int i = 0; i < mNumberOfCameras; i++) { 175 if (mBackCameraId == -1 && mInfo[i].facing == CameraInfo.CAMERA_FACING_BACK) { 176 mBackCameraId = i; 177 } else if (mFrontCameraId == -1 && mInfo[i].facing == CameraInfo.CAMERA_FACING_FRONT) { 178 mFrontCameraId = i; 179 } 180 } 181 } 182 getNumberOfCameras()183 public int getNumberOfCameras() { 184 return mNumberOfCameras; 185 } 186 getCameraInfo()187 public CameraInfo[] getCameraInfo() { 188 return mInfo; 189 } 190 open(int cameraId)191 public synchronized CameraProxy open(int cameraId) 192 throws CameraHardwareException { 193 if (DEBUG_OPEN_RELEASE) { 194 collectState(cameraId, mCameraDevice); 195 if (mCameraOpened) { 196 Log.e(TAG, "double open"); 197 dumpStates(); 198 } 199 } 200 Assert(!mCameraOpened); 201 if (mCameraDevice != null && mCameraId != cameraId) { 202 mCameraDevice.release(); 203 mCameraDevice = null; 204 mCameraId = -1; 205 } 206 if (mCameraDevice == null) { 207 try { 208 Log.v(TAG, "open camera " + cameraId); 209 if (mMockCameraInfo == null) { 210 mCameraDevice = CameraManager.instance().cameraOpen(cameraId); 211 } else { 212 if (mMockCamera == null) 213 throw new RuntimeException(); 214 mCameraDevice = mMockCamera[cameraId]; 215 } 216 mCameraId = cameraId; 217 } catch (RuntimeException e) { 218 Log.e(TAG, "fail to connect Camera", e); 219 throw new CameraHardwareException(e); 220 } 221 mParameters = mCameraDevice.getParameters(); 222 } else { 223 try { 224 mCameraDevice.reconnect(); 225 } catch (IOException e) { 226 Log.e(TAG, "reconnect failed."); 227 throw new CameraHardwareException(e); 228 } 229 mCameraDevice.setParameters(mParameters); 230 } 231 mCameraOpened = true; 232 mHandler.removeMessages(RELEASE_CAMERA); 233 mKeepBeforeTime = 0; 234 return mCameraDevice; 235 } 236 237 /** 238 * Tries to open the hardware camera. If the camera is being used or 239 * unavailable then return {@code null}. 240 */ tryOpen(int cameraId)241 public synchronized CameraProxy tryOpen(int cameraId) { 242 try { 243 return !mCameraOpened ? open(cameraId) : null; 244 } catch (CameraHardwareException e) { 245 // In eng build, we throw the exception so that test tool 246 // can detect it and report it 247 if ("eng".equals(Build.TYPE)) { 248 throw new RuntimeException(e); 249 } 250 return null; 251 } 252 } 253 release()254 public synchronized void release() { 255 if (DEBUG_OPEN_RELEASE) { 256 collectState(mCameraId, mCameraDevice); 257 } 258 259 if (mCameraDevice == null) return; 260 261 long now = System.currentTimeMillis(); 262 if (now < mKeepBeforeTime) { 263 if (mCameraOpened) { 264 mCameraOpened = false; 265 mCameraDevice.stopPreview(); 266 } 267 mHandler.sendEmptyMessageDelayed(RELEASE_CAMERA, 268 mKeepBeforeTime - now); 269 return; 270 } 271 mCameraOpened = false; 272 mCameraDevice.release(); 273 mCameraDevice = null; 274 // We must set this to null because it has a reference to Camera. 275 // Camera has references to the listeners. 276 mParameters = null; 277 mCameraId = -1; 278 } 279 keep()280 public void keep() { 281 keep(KEEP_CAMERA_TIMEOUT); 282 } 283 keep(int time)284 public synchronized void keep(int time) { 285 // We allow mCameraOpened in either state for the convenience of the 286 // calling activity. The activity may not have a chance to call open() 287 // before the user switches to another activity. 288 mKeepBeforeTime = System.currentTimeMillis() + time; 289 } 290 getBackCameraId()291 public int getBackCameraId() { 292 return mBackCameraId; 293 } 294 getFrontCameraId()295 public int getFrontCameraId() { 296 return mFrontCameraId; 297 } 298 } 299