• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2 **
3 ** Copyright (C) 2008, The Android Open Source Project
4 **
5 ** Licensed under the Apache License, Version 2.0 (the "License");
6 ** you may not use this file except in compliance with the License.
7 ** You may obtain a copy of the License at
8 **
9 **     http://www.apache.org/licenses/LICENSE-2.0
10 **
11 ** Unless required by applicable law or agreed to in writing, software
12 ** distributed under the License is distributed on an "AS IS" BASIS,
13 ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 ** See the License for the specific language governing permissions and
15 ** limitations under the License.
16 */
17 
18 //#define LOG_NDEBUG 0
19 #define LOG_TAG "Camera"
20 #include <utils/Log.h>
21 #include <utils/threads.h>
22 #include <utils/String16.h>
23 #include <binder/IPCThreadState.h>
24 #include <binder/IServiceManager.h>
25 #include <binder/IMemory.h>
26 
27 #include <camera/Camera.h>
28 #include <camera/ICameraRecordingProxyListener.h>
29 #include <camera/ICameraService.h>
30 #include <camera/ICamera.h>
31 
32 #include <gui/IGraphicBufferProducer.h>
33 #include <gui/Surface.h>
34 
35 namespace android {
36 
Camera(int cameraId)37 Camera::Camera(int cameraId)
38     : CameraBase(cameraId)
39 {
40 }
41 
42 CameraTraits<Camera>::TCamConnectService CameraTraits<Camera>::fnConnectService =
43         &ICameraService::connect;
44 
45 // construct a camera client from an existing camera remote
create(const sp<ICamera> & camera)46 sp<Camera> Camera::create(const sp<ICamera>& camera)
47 {
48      ALOGV("create");
49      if (camera == 0) {
50          ALOGE("camera remote is a NULL pointer");
51          return 0;
52      }
53 
54     sp<Camera> c = new Camera(-1);
55     if (camera->connect(c) == NO_ERROR) {
56         c->mStatus = NO_ERROR;
57         c->mCamera = camera;
58         camera->asBinder()->linkToDeath(c);
59         return c;
60     }
61     return 0;
62 }
63 
~Camera()64 Camera::~Camera()
65 {
66     // We don't need to call disconnect() here because if the CameraService
67     // thinks we are the owner of the hardware, it will hold a (strong)
68     // reference to us, and we can't possibly be here. We also don't want to
69     // call disconnect() here if we are in the same process as mediaserver,
70     // because we may be invoked by CameraService::Client::connect() and will
71     // deadlock if we call any method of ICamera here.
72 }
73 
connect(int cameraId,const String16 & clientPackageName,int clientUid)74 sp<Camera> Camera::connect(int cameraId, const String16& clientPackageName,
75         int clientUid)
76 {
77     return CameraBaseT::connect(cameraId, clientPackageName, clientUid);
78 }
79 
connectLegacy(int cameraId,int halVersion,const String16 & clientPackageName,int clientUid,sp<Camera> & camera)80 status_t Camera::connectLegacy(int cameraId, int halVersion,
81         const String16& clientPackageName,
82         int clientUid,
83         sp<Camera>& camera)
84 {
85     ALOGV("%s: connect legacy camera device", __FUNCTION__);
86     sp<Camera> c = new Camera(cameraId);
87     sp<ICameraClient> cl = c;
88     status_t status = NO_ERROR;
89     const sp<ICameraService>& cs = CameraBaseT::getCameraService();
90 
91     if (cs != 0) {
92         status = cs.get()->connectLegacy(cl, cameraId, halVersion, clientPackageName,
93                                         clientUid, /*out*/c->mCamera);
94     }
95     if (status == OK && c->mCamera != 0) {
96         c->mCamera->asBinder()->linkToDeath(c);
97         c->mStatus = NO_ERROR;
98         camera = c;
99     } else {
100         ALOGW("An error occurred while connecting to camera: %d", cameraId);
101         c.clear();
102     }
103     return status;
104 }
105 
reconnect()106 status_t Camera::reconnect()
107 {
108     ALOGV("reconnect");
109     sp <ICamera> c = mCamera;
110     if (c == 0) return NO_INIT;
111     return c->connect(this);
112 }
113 
lock()114 status_t Camera::lock()
115 {
116     sp <ICamera> c = mCamera;
117     if (c == 0) return NO_INIT;
118     return c->lock();
119 }
120 
unlock()121 status_t Camera::unlock()
122 {
123     sp <ICamera> c = mCamera;
124     if (c == 0) return NO_INIT;
125     return c->unlock();
126 }
127 
128 // pass the buffered IGraphicBufferProducer to the camera service
setPreviewTarget(const sp<IGraphicBufferProducer> & bufferProducer)129 status_t Camera::setPreviewTarget(const sp<IGraphicBufferProducer>& bufferProducer)
130 {
131     ALOGV("setPreviewTarget(%p)", bufferProducer.get());
132     sp <ICamera> c = mCamera;
133     if (c == 0) return NO_INIT;
134     ALOGD_IF(bufferProducer == 0, "app passed NULL surface");
135     return c->setPreviewTarget(bufferProducer);
136 }
137 
138 // start preview mode
startPreview()139 status_t Camera::startPreview()
140 {
141     ALOGV("startPreview");
142     sp <ICamera> c = mCamera;
143     if (c == 0) return NO_INIT;
144     return c->startPreview();
145 }
146 
storeMetaDataInBuffers(bool enabled)147 status_t Camera::storeMetaDataInBuffers(bool enabled)
148 {
149     ALOGV("storeMetaDataInBuffers: %s",
150             enabled? "true": "false");
151     sp <ICamera> c = mCamera;
152     if (c == 0) return NO_INIT;
153     return c->storeMetaDataInBuffers(enabled);
154 }
155 
156 // start recording mode, must call setPreviewTarget first
startRecording()157 status_t Camera::startRecording()
158 {
159     ALOGV("startRecording");
160     sp <ICamera> c = mCamera;
161     if (c == 0) return NO_INIT;
162     return c->startRecording();
163 }
164 
165 // stop preview mode
stopPreview()166 void Camera::stopPreview()
167 {
168     ALOGV("stopPreview");
169     sp <ICamera> c = mCamera;
170     if (c == 0) return;
171     c->stopPreview();
172 }
173 
174 // stop recording mode
stopRecording()175 void Camera::stopRecording()
176 {
177     ALOGV("stopRecording");
178     {
179         Mutex::Autolock _l(mLock);
180         mRecordingProxyListener.clear();
181     }
182     sp <ICamera> c = mCamera;
183     if (c == 0) return;
184     c->stopRecording();
185 }
186 
187 // release a recording frame
releaseRecordingFrame(const sp<IMemory> & mem)188 void Camera::releaseRecordingFrame(const sp<IMemory>& mem)
189 {
190     ALOGV("releaseRecordingFrame");
191     sp <ICamera> c = mCamera;
192     if (c == 0) return;
193     c->releaseRecordingFrame(mem);
194 }
195 
196 // get preview state
previewEnabled()197 bool Camera::previewEnabled()
198 {
199     ALOGV("previewEnabled");
200     sp <ICamera> c = mCamera;
201     if (c == 0) return false;
202     return c->previewEnabled();
203 }
204 
205 // get recording state
recordingEnabled()206 bool Camera::recordingEnabled()
207 {
208     ALOGV("recordingEnabled");
209     sp <ICamera> c = mCamera;
210     if (c == 0) return false;
211     return c->recordingEnabled();
212 }
213 
autoFocus()214 status_t Camera::autoFocus()
215 {
216     ALOGV("autoFocus");
217     sp <ICamera> c = mCamera;
218     if (c == 0) return NO_INIT;
219     return c->autoFocus();
220 }
221 
cancelAutoFocus()222 status_t Camera::cancelAutoFocus()
223 {
224     ALOGV("cancelAutoFocus");
225     sp <ICamera> c = mCamera;
226     if (c == 0) return NO_INIT;
227     return c->cancelAutoFocus();
228 }
229 
230 // take a picture
takePicture(int msgType)231 status_t Camera::takePicture(int msgType)
232 {
233     ALOGV("takePicture: 0x%x", msgType);
234     sp <ICamera> c = mCamera;
235     if (c == 0) return NO_INIT;
236     return c->takePicture(msgType);
237 }
238 
239 // set preview/capture parameters - key/value pairs
setParameters(const String8 & params)240 status_t Camera::setParameters(const String8& params)
241 {
242     ALOGV("setParameters");
243     sp <ICamera> c = mCamera;
244     if (c == 0) return NO_INIT;
245     return c->setParameters(params);
246 }
247 
248 // get preview/capture parameters - key/value pairs
getParameters() const249 String8 Camera::getParameters() const
250 {
251     ALOGV("getParameters");
252     String8 params;
253     sp <ICamera> c = mCamera;
254     if (c != 0) params = mCamera->getParameters();
255     return params;
256 }
257 
258 // send command to camera driver
sendCommand(int32_t cmd,int32_t arg1,int32_t arg2)259 status_t Camera::sendCommand(int32_t cmd, int32_t arg1, int32_t arg2)
260 {
261     ALOGV("sendCommand");
262     sp <ICamera> c = mCamera;
263     if (c == 0) return NO_INIT;
264     return c->sendCommand(cmd, arg1, arg2);
265 }
266 
setListener(const sp<CameraListener> & listener)267 void Camera::setListener(const sp<CameraListener>& listener)
268 {
269     Mutex::Autolock _l(mLock);
270     mListener = listener;
271 }
272 
setRecordingProxyListener(const sp<ICameraRecordingProxyListener> & listener)273 void Camera::setRecordingProxyListener(const sp<ICameraRecordingProxyListener>& listener)
274 {
275     Mutex::Autolock _l(mLock);
276     mRecordingProxyListener = listener;
277 }
278 
setPreviewCallbackFlags(int flag)279 void Camera::setPreviewCallbackFlags(int flag)
280 {
281     ALOGV("setPreviewCallbackFlags");
282     sp <ICamera> c = mCamera;
283     if (c == 0) return;
284     mCamera->setPreviewCallbackFlag(flag);
285 }
286 
setPreviewCallbackTarget(const sp<IGraphicBufferProducer> & callbackProducer)287 status_t Camera::setPreviewCallbackTarget(
288         const sp<IGraphicBufferProducer>& callbackProducer)
289 {
290     sp <ICamera> c = mCamera;
291     if (c == 0) return NO_INIT;
292     return c->setPreviewCallbackTarget(callbackProducer);
293 }
294 
295 // callback from camera service
notifyCallback(int32_t msgType,int32_t ext1,int32_t ext2)296 void Camera::notifyCallback(int32_t msgType, int32_t ext1, int32_t ext2)
297 {
298     return CameraBaseT::notifyCallback(msgType, ext1, ext2);
299 }
300 
301 // callback from camera service when frame or image is ready
dataCallback(int32_t msgType,const sp<IMemory> & dataPtr,camera_frame_metadata_t * metadata)302 void Camera::dataCallback(int32_t msgType, const sp<IMemory>& dataPtr,
303                           camera_frame_metadata_t *metadata)
304 {
305     sp<CameraListener> listener;
306     {
307         Mutex::Autolock _l(mLock);
308         listener = mListener;
309     }
310     if (listener != NULL) {
311         listener->postData(msgType, dataPtr, metadata);
312     }
313 }
314 
315 // callback from camera service when timestamped frame is ready
dataCallbackTimestamp(nsecs_t timestamp,int32_t msgType,const sp<IMemory> & dataPtr)316 void Camera::dataCallbackTimestamp(nsecs_t timestamp, int32_t msgType, const sp<IMemory>& dataPtr)
317 {
318     // If recording proxy listener is registered, forward the frame and return.
319     // The other listener (mListener) is ignored because the receiver needs to
320     // call releaseRecordingFrame.
321     sp<ICameraRecordingProxyListener> proxylistener;
322     {
323         Mutex::Autolock _l(mLock);
324         proxylistener = mRecordingProxyListener;
325     }
326     if (proxylistener != NULL) {
327         proxylistener->dataCallbackTimestamp(timestamp, msgType, dataPtr);
328         return;
329     }
330 
331     sp<CameraListener> listener;
332     {
333         Mutex::Autolock _l(mLock);
334         listener = mListener;
335     }
336 
337     if (listener != NULL) {
338         listener->postDataTimestamp(timestamp, msgType, dataPtr);
339     } else {
340         ALOGW("No listener was set. Drop a recording frame.");
341         releaseRecordingFrame(dataPtr);
342     }
343 }
344 
getRecordingProxy()345 sp<ICameraRecordingProxy> Camera::getRecordingProxy() {
346     ALOGV("getProxy");
347     return new RecordingProxy(this);
348 }
349 
startRecording(const sp<ICameraRecordingProxyListener> & listener)350 status_t Camera::RecordingProxy::startRecording(const sp<ICameraRecordingProxyListener>& listener)
351 {
352     ALOGV("RecordingProxy::startRecording");
353     mCamera->setRecordingProxyListener(listener);
354     mCamera->reconnect();
355     return mCamera->startRecording();
356 }
357 
stopRecording()358 void Camera::RecordingProxy::stopRecording()
359 {
360     ALOGV("RecordingProxy::stopRecording");
361     mCamera->stopRecording();
362 }
363 
releaseRecordingFrame(const sp<IMemory> & mem)364 void Camera::RecordingProxy::releaseRecordingFrame(const sp<IMemory>& mem)
365 {
366     ALOGV("RecordingProxy::releaseRecordingFrame");
367     mCamera->releaseRecordingFrame(mem);
368 }
369 
RecordingProxy(const sp<Camera> & camera)370 Camera::RecordingProxy::RecordingProxy(const sp<Camera>& camera)
371 {
372     mCamera = camera;
373 }
374 
375 }; // namespace android
376