blob: 1b15a1cbec1a87534cd8c500073fd3b8e59c6029 [file] [log] [blame]
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001/*
2 * Copyright (C) 2012 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
Eino-Ville Talvala4bb81182012-09-24 09:46:53 -070017#define LOG_TAG "Camera2-Parameters"
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -070018#define ATRACE_TAG ATRACE_TAG_CAMERA
19//#define LOG_NDEBUG 0
20
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -070021#include <utils/Log.h>
22#include <utils/Trace.h>
Igor Murashkinaf3d2882012-10-04 14:22:18 -070023#include <utils/Vector.h>
24#include <utils/SortedVector.h>
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -070025
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -070026#include <math.h>
27#include <stdlib.h>
Eino-Ville Talvalab0752522012-09-11 16:51:32 -070028#include <cutils/properties.h>
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -070029
30#include "Parameters.h"
31#include "system/camera.h"
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -070032
33namespace android {
34namespace camera2 {
35
36Parameters::Parameters(int cameraId,
37 int cameraFacing) :
38 cameraId(cameraId),
39 cameraFacing(cameraFacing),
40 info(NULL) {
41}
42
43Parameters::~Parameters() {
44}
45
46status_t Parameters::initialize(const CameraMetadata *info) {
47 status_t res;
48
49 if (info->entryCount() == 0) {
50 ALOGE("%s: No static information provided!", __FUNCTION__);
51 return BAD_VALUE;
52 }
53 Parameters::info = info;
54
55 res = buildFastInfo();
56 if (res != OK) return res;
57
Eino-Ville Talvalae382ee22012-10-02 18:14:49 -070058 res = buildQuirks();
59 if (res != OK) return res;
60
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -070061 camera_metadata_ro_entry_t availableProcessedSizes =
62 staticInfo(ANDROID_SCALER_AVAILABLE_PROCESSED_SIZES, 2);
63 if (!availableProcessedSizes.count) return NO_INIT;
64
65 // TODO: Pick more intelligently
66 previewWidth = availableProcessedSizes.data.i32[0];
67 previewHeight = availableProcessedSizes.data.i32[1];
68 videoWidth = previewWidth;
69 videoHeight = previewHeight;
70
71 params.setPreviewSize(previewWidth, previewHeight);
72 params.setVideoSize(videoWidth, videoHeight);
73 params.set(CameraParameters::KEY_PREFERRED_PREVIEW_SIZE_FOR_VIDEO,
74 String8::format("%dx%d",
75 previewWidth, previewHeight));
76 {
77 String8 supportedPreviewSizes;
78 for (size_t i=0; i < availableProcessedSizes.count; i += 2) {
79 if (i != 0) supportedPreviewSizes += ",";
80 supportedPreviewSizes += String8::format("%dx%d",
81 availableProcessedSizes.data.i32[i],
82 availableProcessedSizes.data.i32[i+1]);
83 }
84 params.set(CameraParameters::KEY_SUPPORTED_PREVIEW_SIZES,
85 supportedPreviewSizes);
86 params.set(CameraParameters::KEY_SUPPORTED_VIDEO_SIZES,
87 supportedPreviewSizes);
88 }
89
90 camera_metadata_ro_entry_t availableFpsRanges =
91 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_TARGET_FPS_RANGES, 2);
92 if (!availableFpsRanges.count) return NO_INIT;
93
94 previewFpsRange[0] = availableFpsRanges.data.i32[0];
95 previewFpsRange[1] = availableFpsRanges.data.i32[1];
96
97 params.set(CameraParameters::KEY_PREVIEW_FPS_RANGE,
98 String8::format("%d,%d",
Eino-Ville Talvalac9d7e4d2012-09-27 14:18:13 -070099 previewFpsRange[0] * kFpsToApiScale,
100 previewFpsRange[1] * kFpsToApiScale));
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700101
102 {
103 String8 supportedPreviewFpsRange;
104 for (size_t i=0; i < availableFpsRanges.count; i += 2) {
105 if (i != 0) supportedPreviewFpsRange += ",";
106 supportedPreviewFpsRange += String8::format("(%d,%d)",
Eino-Ville Talvalac9d7e4d2012-09-27 14:18:13 -0700107 availableFpsRanges.data.i32[i] * kFpsToApiScale,
108 availableFpsRanges.data.i32[i+1] * kFpsToApiScale);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700109 }
110 params.set(CameraParameters::KEY_SUPPORTED_PREVIEW_FPS_RANGE,
111 supportedPreviewFpsRange);
112 }
113
114 previewFormat = HAL_PIXEL_FORMAT_YCrCb_420_SP;
115 params.set(CameraParameters::KEY_PREVIEW_FORMAT,
116 formatEnumToString(previewFormat)); // NV21
117
118 previewTransform = degToTransform(0,
119 cameraFacing == CAMERA_FACING_FRONT);
120
121 camera_metadata_ro_entry_t availableFormats =
122 staticInfo(ANDROID_SCALER_AVAILABLE_FORMATS);
123
124 {
125 String8 supportedPreviewFormats;
126 bool addComma = false;
127 for (size_t i=0; i < availableFormats.count; i++) {
128 if (addComma) supportedPreviewFormats += ",";
129 addComma = true;
130 switch (availableFormats.data.i32[i]) {
131 case HAL_PIXEL_FORMAT_YCbCr_422_SP:
132 supportedPreviewFormats +=
133 CameraParameters::PIXEL_FORMAT_YUV422SP;
134 break;
135 case HAL_PIXEL_FORMAT_YCrCb_420_SP:
136 supportedPreviewFormats +=
137 CameraParameters::PIXEL_FORMAT_YUV420SP;
138 break;
139 case HAL_PIXEL_FORMAT_YCbCr_422_I:
140 supportedPreviewFormats +=
141 CameraParameters::PIXEL_FORMAT_YUV422I;
142 break;
143 case HAL_PIXEL_FORMAT_YV12:
144 supportedPreviewFormats +=
145 CameraParameters::PIXEL_FORMAT_YUV420P;
146 break;
147 case HAL_PIXEL_FORMAT_RGB_565:
148 supportedPreviewFormats +=
149 CameraParameters::PIXEL_FORMAT_RGB565;
150 break;
151 case HAL_PIXEL_FORMAT_RGBA_8888:
152 supportedPreviewFormats +=
153 CameraParameters::PIXEL_FORMAT_RGBA8888;
154 break;
155 // Not advertizing JPEG, RAW_SENSOR, etc, for preview formats
156 case HAL_PIXEL_FORMAT_RAW_SENSOR:
157 case HAL_PIXEL_FORMAT_BLOB:
158 addComma = false;
159 break;
160
161 default:
162 ALOGW("%s: Camera %d: Unknown preview format: %x",
163 __FUNCTION__, cameraId, availableFormats.data.i32[i]);
164 addComma = false;
165 break;
166 }
167 }
168 params.set(CameraParameters::KEY_SUPPORTED_PREVIEW_FORMATS,
169 supportedPreviewFormats);
170 }
171
172 // PREVIEW_FRAME_RATE / SUPPORTED_PREVIEW_FRAME_RATES are deprecated, but
173 // still have to do something sane for them
174
Eino-Ville Talvalad0cec0c2012-09-27 18:08:20 -0700175 // NOTE: Not scaled like FPS range values are.
Igor Murashkinaf3d2882012-10-04 14:22:18 -0700176 previewFps = fpsFromRange(previewFpsRange[0], previewFpsRange[1]);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700177 params.set(CameraParameters::KEY_PREVIEW_FRAME_RATE,
Igor Murashkinaf3d2882012-10-04 14:22:18 -0700178 previewFps);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700179
180 {
Igor Murashkinaf3d2882012-10-04 14:22:18 -0700181 SortedVector<int32_t> sortedPreviewFrameRates;
182
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700183 String8 supportedPreviewFrameRates;
184 for (size_t i=0; i < availableFpsRanges.count; i += 2) {
Igor Murashkinaf3d2882012-10-04 14:22:18 -0700185 // from the [min, max] fps range use the max value
186 int fps = fpsFromRange(availableFpsRanges.data.i32[i],
187 availableFpsRanges.data.i32[i+1]);
188
189 // de-dupe frame rates
190 if (sortedPreviewFrameRates.indexOf(fps) == NAME_NOT_FOUND) {
191 sortedPreviewFrameRates.add(fps);
192 }
193 else {
194 continue;
195 }
196
197 if (sortedPreviewFrameRates.size() > 1) {
198 supportedPreviewFrameRates += ",";
199 }
200
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700201 supportedPreviewFrameRates += String8::format("%d",
Igor Murashkinaf3d2882012-10-04 14:22:18 -0700202 fps);
203
204 ALOGV("%s: Supported preview frame rates: %s",
205 __FUNCTION__, supportedPreviewFrameRates.string());
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700206 }
207 params.set(CameraParameters::KEY_SUPPORTED_PREVIEW_FRAME_RATES,
208 supportedPreviewFrameRates);
209 }
210
211 camera_metadata_ro_entry_t availableJpegSizes =
212 staticInfo(ANDROID_SCALER_AVAILABLE_JPEG_SIZES, 2);
213 if (!availableJpegSizes.count) return NO_INIT;
214
215 // TODO: Pick maximum
216 pictureWidth = availableJpegSizes.data.i32[0];
217 pictureHeight = availableJpegSizes.data.i32[1];
218
219 params.setPictureSize(pictureWidth,
220 pictureHeight);
221
222 {
223 String8 supportedPictureSizes;
224 for (size_t i=0; i < availableJpegSizes.count; i += 2) {
225 if (i != 0) supportedPictureSizes += ",";
226 supportedPictureSizes += String8::format("%dx%d",
227 availableJpegSizes.data.i32[i],
228 availableJpegSizes.data.i32[i+1]);
229 }
230 params.set(CameraParameters::KEY_SUPPORTED_PICTURE_SIZES,
231 supportedPictureSizes);
232 }
233
234 params.setPictureFormat(CameraParameters::PIXEL_FORMAT_JPEG);
235 params.set(CameraParameters::KEY_SUPPORTED_PICTURE_FORMATS,
236 CameraParameters::PIXEL_FORMAT_JPEG);
237
238 camera_metadata_ro_entry_t availableJpegThumbnailSizes =
Eino-Ville Talvalaf128dc52012-08-28 13:38:23 -0700239 staticInfo(ANDROID_JPEG_AVAILABLE_THUMBNAIL_SIZES, 4);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700240 if (!availableJpegThumbnailSizes.count) return NO_INIT;
241
242 // TODO: Pick default thumbnail size sensibly
243 jpegThumbSize[0] = availableJpegThumbnailSizes.data.i32[0];
244 jpegThumbSize[1] = availableJpegThumbnailSizes.data.i32[1];
245
246 params.set(CameraParameters::KEY_JPEG_THUMBNAIL_WIDTH,
247 jpegThumbSize[0]);
248 params.set(CameraParameters::KEY_JPEG_THUMBNAIL_HEIGHT,
249 jpegThumbSize[1]);
250
251 {
252 String8 supportedJpegThumbSizes;
253 for (size_t i=0; i < availableJpegThumbnailSizes.count; i += 2) {
254 if (i != 0) supportedJpegThumbSizes += ",";
255 supportedJpegThumbSizes += String8::format("%dx%d",
256 availableJpegThumbnailSizes.data.i32[i],
257 availableJpegThumbnailSizes.data.i32[i+1]);
258 }
259 params.set(CameraParameters::KEY_SUPPORTED_JPEG_THUMBNAIL_SIZES,
260 supportedJpegThumbSizes);
261 }
262
263 jpegThumbQuality = 90;
264 params.set(CameraParameters::KEY_JPEG_THUMBNAIL_QUALITY,
265 jpegThumbQuality);
266 jpegQuality = 90;
267 params.set(CameraParameters::KEY_JPEG_QUALITY,
268 jpegQuality);
269 jpegRotation = 0;
270 params.set(CameraParameters::KEY_ROTATION,
271 jpegRotation);
272
273 gpsEnabled = false;
274 gpsProcessingMethod = "unknown";
275 // GPS fields in CameraParameters are not set by implementation
276
277 wbMode = ANDROID_CONTROL_AWB_AUTO;
278 params.set(CameraParameters::KEY_WHITE_BALANCE,
279 CameraParameters::WHITE_BALANCE_AUTO);
280
281 camera_metadata_ro_entry_t availableWhiteBalanceModes =
282 staticInfo(ANDROID_CONTROL_AWB_AVAILABLE_MODES);
283 {
284 String8 supportedWhiteBalance;
285 bool addComma = false;
286 for (size_t i=0; i < availableWhiteBalanceModes.count; i++) {
287 if (addComma) supportedWhiteBalance += ",";
288 addComma = true;
289 switch (availableWhiteBalanceModes.data.u8[i]) {
290 case ANDROID_CONTROL_AWB_AUTO:
291 supportedWhiteBalance +=
292 CameraParameters::WHITE_BALANCE_AUTO;
293 break;
294 case ANDROID_CONTROL_AWB_INCANDESCENT:
295 supportedWhiteBalance +=
296 CameraParameters::WHITE_BALANCE_INCANDESCENT;
297 break;
298 case ANDROID_CONTROL_AWB_FLUORESCENT:
299 supportedWhiteBalance +=
300 CameraParameters::WHITE_BALANCE_FLUORESCENT;
301 break;
302 case ANDROID_CONTROL_AWB_WARM_FLUORESCENT:
303 supportedWhiteBalance +=
304 CameraParameters::WHITE_BALANCE_WARM_FLUORESCENT;
305 break;
306 case ANDROID_CONTROL_AWB_DAYLIGHT:
307 supportedWhiteBalance +=
308 CameraParameters::WHITE_BALANCE_DAYLIGHT;
309 break;
310 case ANDROID_CONTROL_AWB_CLOUDY_DAYLIGHT:
311 supportedWhiteBalance +=
312 CameraParameters::WHITE_BALANCE_CLOUDY_DAYLIGHT;
313 break;
314 case ANDROID_CONTROL_AWB_TWILIGHT:
315 supportedWhiteBalance +=
316 CameraParameters::WHITE_BALANCE_TWILIGHT;
317 break;
318 case ANDROID_CONTROL_AWB_SHADE:
319 supportedWhiteBalance +=
320 CameraParameters::WHITE_BALANCE_SHADE;
321 break;
322 // Skipping values not mappable to v1 API
323 case ANDROID_CONTROL_AWB_OFF:
324 addComma = false;
325 break;
326 default:
327 ALOGW("%s: Camera %d: Unknown white balance value: %d",
328 __FUNCTION__, cameraId,
329 availableWhiteBalanceModes.data.u8[i]);
330 addComma = false;
331 break;
332 }
333 }
334 params.set(CameraParameters::KEY_SUPPORTED_WHITE_BALANCE,
335 supportedWhiteBalance);
336 }
337
338 effectMode = ANDROID_CONTROL_EFFECT_OFF;
339 params.set(CameraParameters::KEY_EFFECT,
340 CameraParameters::EFFECT_NONE);
341
342 camera_metadata_ro_entry_t availableEffects =
343 staticInfo(ANDROID_CONTROL_AVAILABLE_EFFECTS);
344 if (!availableEffects.count) return NO_INIT;
345 {
346 String8 supportedEffects;
347 bool addComma = false;
348 for (size_t i=0; i < availableEffects.count; i++) {
349 if (addComma) supportedEffects += ",";
350 addComma = true;
351 switch (availableEffects.data.u8[i]) {
352 case ANDROID_CONTROL_EFFECT_OFF:
353 supportedEffects +=
354 CameraParameters::EFFECT_NONE;
355 break;
356 case ANDROID_CONTROL_EFFECT_MONO:
357 supportedEffects +=
358 CameraParameters::EFFECT_MONO;
359 break;
360 case ANDROID_CONTROL_EFFECT_NEGATIVE:
361 supportedEffects +=
362 CameraParameters::EFFECT_NEGATIVE;
363 break;
364 case ANDROID_CONTROL_EFFECT_SOLARIZE:
365 supportedEffects +=
366 CameraParameters::EFFECT_SOLARIZE;
367 break;
368 case ANDROID_CONTROL_EFFECT_SEPIA:
369 supportedEffects +=
370 CameraParameters::EFFECT_SEPIA;
371 break;
372 case ANDROID_CONTROL_EFFECT_POSTERIZE:
373 supportedEffects +=
374 CameraParameters::EFFECT_POSTERIZE;
375 break;
376 case ANDROID_CONTROL_EFFECT_WHITEBOARD:
377 supportedEffects +=
378 CameraParameters::EFFECT_WHITEBOARD;
379 break;
380 case ANDROID_CONTROL_EFFECT_BLACKBOARD:
381 supportedEffects +=
382 CameraParameters::EFFECT_BLACKBOARD;
383 break;
384 case ANDROID_CONTROL_EFFECT_AQUA:
385 supportedEffects +=
386 CameraParameters::EFFECT_AQUA;
387 break;
388 default:
389 ALOGW("%s: Camera %d: Unknown effect value: %d",
390 __FUNCTION__, cameraId, availableEffects.data.u8[i]);
391 addComma = false;
392 break;
393 }
394 }
395 params.set(CameraParameters::KEY_SUPPORTED_EFFECTS, supportedEffects);
396 }
397
398 antibandingMode = ANDROID_CONTROL_AE_ANTIBANDING_AUTO;
399 params.set(CameraParameters::KEY_ANTIBANDING,
400 CameraParameters::ANTIBANDING_AUTO);
401
402 camera_metadata_ro_entry_t availableAntibandingModes =
403 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_ANTIBANDING_MODES);
404 if (!availableAntibandingModes.count) return NO_INIT;
405 {
406 String8 supportedAntibanding;
407 bool addComma = false;
408 for (size_t i=0; i < availableAntibandingModes.count; i++) {
409 if (addComma) supportedAntibanding += ",";
410 addComma = true;
411 switch (availableAntibandingModes.data.u8[i]) {
412 case ANDROID_CONTROL_AE_ANTIBANDING_OFF:
413 supportedAntibanding +=
414 CameraParameters::ANTIBANDING_OFF;
415 break;
416 case ANDROID_CONTROL_AE_ANTIBANDING_50HZ:
417 supportedAntibanding +=
418 CameraParameters::ANTIBANDING_50HZ;
419 break;
420 case ANDROID_CONTROL_AE_ANTIBANDING_60HZ:
421 supportedAntibanding +=
422 CameraParameters::ANTIBANDING_60HZ;
423 break;
424 case ANDROID_CONTROL_AE_ANTIBANDING_AUTO:
425 supportedAntibanding +=
426 CameraParameters::ANTIBANDING_AUTO;
427 break;
428 default:
429 ALOGW("%s: Camera %d: Unknown antibanding value: %d",
430 __FUNCTION__, cameraId,
431 availableAntibandingModes.data.u8[i]);
432 addComma = false;
433 break;
434 }
435 }
436 params.set(CameraParameters::KEY_SUPPORTED_ANTIBANDING,
437 supportedAntibanding);
438 }
439
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -0700440 sceneMode = ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700441 params.set(CameraParameters::KEY_SCENE_MODE,
442 CameraParameters::SCENE_MODE_AUTO);
443
444 camera_metadata_ro_entry_t availableSceneModes =
445 staticInfo(ANDROID_CONTROL_AVAILABLE_SCENE_MODES);
446 if (!availableSceneModes.count) return NO_INIT;
447 {
448 String8 supportedSceneModes(CameraParameters::SCENE_MODE_AUTO);
449 bool addComma = true;
450 bool noSceneModes = false;
451 for (size_t i=0; i < availableSceneModes.count; i++) {
452 if (addComma) supportedSceneModes += ",";
453 addComma = true;
454 switch (availableSceneModes.data.u8[i]) {
455 case ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED:
456 noSceneModes = true;
457 break;
458 case ANDROID_CONTROL_SCENE_MODE_FACE_PRIORITY:
459 // Not in old API
460 addComma = false;
461 break;
462 case ANDROID_CONTROL_SCENE_MODE_ACTION:
463 supportedSceneModes +=
464 CameraParameters::SCENE_MODE_ACTION;
465 break;
466 case ANDROID_CONTROL_SCENE_MODE_PORTRAIT:
467 supportedSceneModes +=
468 CameraParameters::SCENE_MODE_PORTRAIT;
469 break;
470 case ANDROID_CONTROL_SCENE_MODE_LANDSCAPE:
471 supportedSceneModes +=
472 CameraParameters::SCENE_MODE_LANDSCAPE;
473 break;
474 case ANDROID_CONTROL_SCENE_MODE_NIGHT:
475 supportedSceneModes +=
476 CameraParameters::SCENE_MODE_NIGHT;
477 break;
478 case ANDROID_CONTROL_SCENE_MODE_NIGHT_PORTRAIT:
479 supportedSceneModes +=
480 CameraParameters::SCENE_MODE_NIGHT_PORTRAIT;
481 break;
482 case ANDROID_CONTROL_SCENE_MODE_THEATRE:
483 supportedSceneModes +=
484 CameraParameters::SCENE_MODE_THEATRE;
485 break;
486 case ANDROID_CONTROL_SCENE_MODE_BEACH:
487 supportedSceneModes +=
488 CameraParameters::SCENE_MODE_BEACH;
489 break;
490 case ANDROID_CONTROL_SCENE_MODE_SNOW:
491 supportedSceneModes +=
492 CameraParameters::SCENE_MODE_SNOW;
493 break;
494 case ANDROID_CONTROL_SCENE_MODE_SUNSET:
495 supportedSceneModes +=
496 CameraParameters::SCENE_MODE_SUNSET;
497 break;
498 case ANDROID_CONTROL_SCENE_MODE_STEADYPHOTO:
499 supportedSceneModes +=
500 CameraParameters::SCENE_MODE_STEADYPHOTO;
501 break;
502 case ANDROID_CONTROL_SCENE_MODE_FIREWORKS:
503 supportedSceneModes +=
504 CameraParameters::SCENE_MODE_FIREWORKS;
505 break;
506 case ANDROID_CONTROL_SCENE_MODE_SPORTS:
507 supportedSceneModes +=
508 CameraParameters::SCENE_MODE_SPORTS;
509 break;
510 case ANDROID_CONTROL_SCENE_MODE_PARTY:
511 supportedSceneModes +=
512 CameraParameters::SCENE_MODE_PARTY;
513 break;
514 case ANDROID_CONTROL_SCENE_MODE_CANDLELIGHT:
515 supportedSceneModes +=
516 CameraParameters::SCENE_MODE_CANDLELIGHT;
517 break;
518 case ANDROID_CONTROL_SCENE_MODE_BARCODE:
519 supportedSceneModes +=
520 CameraParameters::SCENE_MODE_BARCODE;
521 break;
522 default:
523 ALOGW("%s: Camera %d: Unknown scene mode value: %d",
524 __FUNCTION__, cameraId,
525 availableSceneModes.data.u8[i]);
526 addComma = false;
527 break;
528 }
529 }
530 if (!noSceneModes) {
531 params.set(CameraParameters::KEY_SUPPORTED_SCENE_MODES,
532 supportedSceneModes);
533 }
534 }
535
536 camera_metadata_ro_entry_t flashAvailable =
537 staticInfo(ANDROID_FLASH_AVAILABLE, 1, 1);
538 if (!flashAvailable.count) return NO_INIT;
539
540 camera_metadata_ro_entry_t availableAeModes =
541 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_MODES);
542 if (!availableAeModes.count) return NO_INIT;
543
544 if (flashAvailable.data.u8[0]) {
545 flashMode = Parameters::FLASH_MODE_AUTO;
546 params.set(CameraParameters::KEY_FLASH_MODE,
547 CameraParameters::FLASH_MODE_AUTO);
548
549 String8 supportedFlashModes(CameraParameters::FLASH_MODE_OFF);
550 supportedFlashModes = supportedFlashModes +
551 "," + CameraParameters::FLASH_MODE_AUTO +
552 "," + CameraParameters::FLASH_MODE_ON +
553 "," + CameraParameters::FLASH_MODE_TORCH;
554 for (size_t i=0; i < availableAeModes.count; i++) {
555 if (availableAeModes.data.u8[i] ==
556 ANDROID_CONTROL_AE_ON_AUTO_FLASH_REDEYE) {
557 supportedFlashModes = supportedFlashModes + "," +
558 CameraParameters::FLASH_MODE_RED_EYE;
559 break;
560 }
561 }
562 params.set(CameraParameters::KEY_SUPPORTED_FLASH_MODES,
563 supportedFlashModes);
564 } else {
565 flashMode = Parameters::FLASH_MODE_OFF;
566 params.set(CameraParameters::KEY_FLASH_MODE,
567 CameraParameters::FLASH_MODE_OFF);
568 params.set(CameraParameters::KEY_SUPPORTED_FLASH_MODES,
569 CameraParameters::FLASH_MODE_OFF);
570 }
571
572 camera_metadata_ro_entry_t minFocusDistance =
573 staticInfo(ANDROID_LENS_MINIMUM_FOCUS_DISTANCE, 1, 1);
574 if (!minFocusDistance.count) return NO_INIT;
575
576 camera_metadata_ro_entry_t availableAfModes =
577 staticInfo(ANDROID_CONTROL_AF_AVAILABLE_MODES);
578 if (!availableAfModes.count) return NO_INIT;
579
580 if (minFocusDistance.data.f[0] == 0) {
581 // Fixed-focus lens
582 focusMode = Parameters::FOCUS_MODE_FIXED;
583 params.set(CameraParameters::KEY_FOCUS_MODE,
584 CameraParameters::FOCUS_MODE_FIXED);
585 params.set(CameraParameters::KEY_SUPPORTED_FOCUS_MODES,
586 CameraParameters::FOCUS_MODE_FIXED);
587 } else {
588 focusMode = Parameters::FOCUS_MODE_AUTO;
589 params.set(CameraParameters::KEY_FOCUS_MODE,
590 CameraParameters::FOCUS_MODE_AUTO);
591 String8 supportedFocusModes(CameraParameters::FOCUS_MODE_INFINITY);
592 bool addComma = true;
593
594 for (size_t i=0; i < availableAfModes.count; i++) {
595 if (addComma) supportedFocusModes += ",";
596 addComma = true;
597 switch (availableAfModes.data.u8[i]) {
598 case ANDROID_CONTROL_AF_AUTO:
599 supportedFocusModes +=
600 CameraParameters::FOCUS_MODE_AUTO;
601 break;
602 case ANDROID_CONTROL_AF_MACRO:
603 supportedFocusModes +=
604 CameraParameters::FOCUS_MODE_MACRO;
605 break;
606 case ANDROID_CONTROL_AF_CONTINUOUS_VIDEO:
607 supportedFocusModes +=
608 CameraParameters::FOCUS_MODE_CONTINUOUS_VIDEO;
609 break;
610 case ANDROID_CONTROL_AF_CONTINUOUS_PICTURE:
611 supportedFocusModes +=
612 CameraParameters::FOCUS_MODE_CONTINUOUS_PICTURE;
613 break;
614 case ANDROID_CONTROL_AF_EDOF:
615 supportedFocusModes +=
616 CameraParameters::FOCUS_MODE_EDOF;
617 break;
618 // Not supported in old API
619 case ANDROID_CONTROL_AF_OFF:
620 addComma = false;
621 break;
622 default:
623 ALOGW("%s: Camera %d: Unknown AF mode value: %d",
624 __FUNCTION__, cameraId, availableAfModes.data.u8[i]);
625 addComma = false;
626 break;
627 }
628 }
629 params.set(CameraParameters::KEY_SUPPORTED_FOCUS_MODES,
630 supportedFocusModes);
631 }
Eino-Ville Talvala95069fe2012-10-04 00:56:40 -0700632 shadowFocusMode = FOCUS_MODE_INVALID;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700633
634 camera_metadata_ro_entry_t max3aRegions =
635 staticInfo(ANDROID_CONTROL_MAX_REGIONS, 1, 1);
636 if (!max3aRegions.count) return NO_INIT;
637
638 params.set(CameraParameters::KEY_MAX_NUM_FOCUS_AREAS,
639 max3aRegions.data.i32[0]);
640 params.set(CameraParameters::KEY_FOCUS_AREAS,
641 "(0,0,0,0,0)");
642 focusingAreas.clear();
643 focusingAreas.add(Parameters::Area(0,0,0,0,0));
644
645 camera_metadata_ro_entry_t availableFocalLengths =
646 staticInfo(ANDROID_LENS_AVAILABLE_FOCAL_LENGTHS);
647 if (!availableFocalLengths.count) return NO_INIT;
648
649 float minFocalLength = availableFocalLengths.data.f[0];
650 params.setFloat(CameraParameters::KEY_FOCAL_LENGTH, minFocalLength);
651
652 camera_metadata_ro_entry_t sensorSize =
653 staticInfo(ANDROID_SENSOR_PHYSICAL_SIZE, 2, 2);
654 if (!sensorSize.count) return NO_INIT;
655
656 // The fields of view here assume infinity focus, maximum wide angle
657 float horizFov = 180 / M_PI *
658 2 * atanf(sensorSize.data.f[0] / (2 * minFocalLength));
659 float vertFov = 180 / M_PI *
660 2 * atanf(sensorSize.data.f[1] / (2 * minFocalLength));
661 params.setFloat(CameraParameters::KEY_HORIZONTAL_VIEW_ANGLE, horizFov);
662 params.setFloat(CameraParameters::KEY_VERTICAL_VIEW_ANGLE, vertFov);
663
664 exposureCompensation = 0;
665 params.set(CameraParameters::KEY_EXPOSURE_COMPENSATION,
666 exposureCompensation);
667
668 camera_metadata_ro_entry_t exposureCompensationRange =
669 staticInfo(ANDROID_CONTROL_AE_EXP_COMPENSATION_RANGE, 2, 2);
670 if (!exposureCompensationRange.count) return NO_INIT;
671
672 params.set(CameraParameters::KEY_MAX_EXPOSURE_COMPENSATION,
673 exposureCompensationRange.data.i32[1]);
674 params.set(CameraParameters::KEY_MIN_EXPOSURE_COMPENSATION,
675 exposureCompensationRange.data.i32[0]);
676
677 camera_metadata_ro_entry_t exposureCompensationStep =
678 staticInfo(ANDROID_CONTROL_AE_EXP_COMPENSATION_STEP, 1, 1);
679 if (!exposureCompensationStep.count) return NO_INIT;
680
681 params.setFloat(CameraParameters::KEY_EXPOSURE_COMPENSATION_STEP,
682 (float)exposureCompensationStep.data.r[0].numerator /
683 exposureCompensationStep.data.r[0].denominator);
684
685 autoExposureLock = false;
686 params.set(CameraParameters::KEY_AUTO_EXPOSURE_LOCK,
687 CameraParameters::FALSE);
688 params.set(CameraParameters::KEY_AUTO_EXPOSURE_LOCK_SUPPORTED,
689 CameraParameters::TRUE);
690
691 autoWhiteBalanceLock = false;
692 params.set(CameraParameters::KEY_AUTO_WHITEBALANCE_LOCK,
693 CameraParameters::FALSE);
694 params.set(CameraParameters::KEY_AUTO_WHITEBALANCE_LOCK_SUPPORTED,
695 CameraParameters::TRUE);
696
697 meteringAreas.add(Parameters::Area(0, 0, 0, 0, 0));
698 params.set(CameraParameters::KEY_MAX_NUM_METERING_AREAS,
699 max3aRegions.data.i32[0]);
700 params.set(CameraParameters::KEY_METERING_AREAS,
701 "(0,0,0,0,0)");
702
703 zoom = 0;
704 params.set(CameraParameters::KEY_ZOOM, zoom);
705 params.set(CameraParameters::KEY_MAX_ZOOM, NUM_ZOOM_STEPS - 1);
706
707 camera_metadata_ro_entry_t maxDigitalZoom =
Igor Murashkin018d2282012-09-18 18:23:49 -0700708 staticInfo(ANDROID_SCALER_AVAILABLE_MAX_ZOOM, /*minCount*/1, /*maxCount*/1);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700709 if (!maxDigitalZoom.count) return NO_INIT;
710
711 {
712 String8 zoomRatios;
713 float zoom = 1.f;
714 float zoomIncrement = (maxDigitalZoom.data.f[0] - zoom) /
715 (NUM_ZOOM_STEPS-1);
716 bool addComma = false;
717 for (size_t i=0; i < NUM_ZOOM_STEPS; i++) {
718 if (addComma) zoomRatios += ",";
719 addComma = true;
720 zoomRatios += String8::format("%d", static_cast<int>(zoom * 100));
721 zoom += zoomIncrement;
722 }
723 params.set(CameraParameters::KEY_ZOOM_RATIOS, zoomRatios);
724 }
725
726 params.set(CameraParameters::KEY_ZOOM_SUPPORTED,
727 CameraParameters::TRUE);
728 params.set(CameraParameters::KEY_SMOOTH_ZOOM_SUPPORTED,
729 CameraParameters::TRUE);
730
731 params.set(CameraParameters::KEY_FOCUS_DISTANCES,
732 "Infinity,Infinity,Infinity");
733
734 params.set(CameraParameters::KEY_MAX_NUM_DETECTED_FACES_HW,
735 fastInfo.maxFaces);
736 params.set(CameraParameters::KEY_MAX_NUM_DETECTED_FACES_SW,
737 0);
738
739 params.set(CameraParameters::KEY_VIDEO_FRAME_FORMAT,
740 CameraParameters::PIXEL_FORMAT_ANDROID_OPAQUE);
741
742 params.set(CameraParameters::KEY_RECORDING_HINT,
743 CameraParameters::FALSE);
744
745 params.set(CameraParameters::KEY_VIDEO_SNAPSHOT_SUPPORTED,
746 CameraParameters::TRUE);
747
748 params.set(CameraParameters::KEY_VIDEO_STABILIZATION,
749 CameraParameters::FALSE);
750
751 camera_metadata_ro_entry_t availableVideoStabilizationModes =
752 staticInfo(ANDROID_CONTROL_AVAILABLE_VIDEO_STABILIZATION_MODES);
753 if (!availableVideoStabilizationModes.count) return NO_INIT;
754
755 if (availableVideoStabilizationModes.count > 1) {
756 params.set(CameraParameters::KEY_VIDEO_STABILIZATION_SUPPORTED,
757 CameraParameters::TRUE);
758 } else {
759 params.set(CameraParameters::KEY_VIDEO_STABILIZATION_SUPPORTED,
760 CameraParameters::FALSE);
761 }
762
763 // Set up initial state for non-Camera.Parameters state variables
764
765 storeMetadataInBuffers = true;
766 playShutterSound = true;
767 enableFaceDetect = false;
768
769 enableFocusMoveMessages = false;
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -0700770 afTriggerCounter = 1;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700771 currentAfTriggerId = -1;
772
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -0700773 precaptureTriggerCounter = 1;
774
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700775 previewCallbackFlags = 0;
776
Eino-Ville Talvalab0752522012-09-11 16:51:32 -0700777 char value[PROPERTY_VALUE_MAX];
Eino-Ville Talvala2f931272012-09-17 16:57:27 -0700778 property_get("camera.disable_zsl_mode", value, "0");
Eino-Ville Talvalab0752522012-09-11 16:51:32 -0700779 if (!strcmp(value,"1")) {
Eino-Ville Talvala2f931272012-09-17 16:57:27 -0700780 ALOGI("Camera %d: Disabling ZSL mode", cameraId);
Eino-Ville Talvalab0752522012-09-11 16:51:32 -0700781 zslMode = false;
Eino-Ville Talvala2f931272012-09-17 16:57:27 -0700782 } else {
783 zslMode = true;
Eino-Ville Talvalab0752522012-09-11 16:51:32 -0700784 }
James Paintere5382062012-09-05 18:02:32 -0700785
786 lightFx = LIGHTFX_NONE;
787
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700788 state = STOPPED;
789
790 paramsFlattened = params.flatten();
791
792 return OK;
793}
794
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -0700795String8 Parameters::get() const {
796 return paramsFlattened;
797}
798
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700799status_t Parameters::buildFastInfo() {
800
801 camera_metadata_ro_entry_t activeArraySize =
802 staticInfo(ANDROID_SENSOR_ACTIVE_ARRAY_SIZE, 2, 2);
803 if (!activeArraySize.count) return NO_INIT;
804 int32_t arrayWidth = activeArraySize.data.i32[0];
805 int32_t arrayHeight = activeArraySize.data.i32[1];
806
807 camera_metadata_ro_entry_t availableFaceDetectModes =
808 staticInfo(ANDROID_STATS_AVAILABLE_FACE_DETECT_MODES);
809 if (!availableFaceDetectModes.count) return NO_INIT;
810
811 uint8_t bestFaceDetectMode =
812 ANDROID_STATS_FACE_DETECTION_OFF;
813 for (size_t i = 0 ; i < availableFaceDetectModes.count; i++) {
814 switch (availableFaceDetectModes.data.u8[i]) {
815 case ANDROID_STATS_FACE_DETECTION_OFF:
816 break;
817 case ANDROID_STATS_FACE_DETECTION_SIMPLE:
818 if (bestFaceDetectMode !=
819 ANDROID_STATS_FACE_DETECTION_FULL) {
820 bestFaceDetectMode =
821 ANDROID_STATS_FACE_DETECTION_SIMPLE;
822 }
823 break;
824 case ANDROID_STATS_FACE_DETECTION_FULL:
825 bestFaceDetectMode =
826 ANDROID_STATS_FACE_DETECTION_FULL;
827 break;
828 default:
829 ALOGE("%s: Camera %d: Unknown face detect mode %d:",
830 __FUNCTION__, cameraId,
831 availableFaceDetectModes.data.u8[i]);
832 return NO_INIT;
833 }
834 }
835
836 camera_metadata_ro_entry_t maxFacesDetected =
837 staticInfo(ANDROID_STATS_MAX_FACE_COUNT, 1, 1);
838 if (!maxFacesDetected.count) return NO_INIT;
839
840 int32_t maxFaces = maxFacesDetected.data.i32[0];
841
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -0700842 camera_metadata_ro_entry_t availableSceneModes =
843 staticInfo(ANDROID_CONTROL_AVAILABLE_SCENE_MODES);
844 camera_metadata_ro_entry_t sceneModeOverrides =
845 staticInfo(ANDROID_CONTROL_SCENE_MODE_OVERRIDES);
846 camera_metadata_ro_entry_t minFocusDistance =
847 staticInfo(ANDROID_LENS_MINIMUM_FOCUS_DISTANCE);
848 bool fixedLens = (minFocusDistance.data.f[0] == 0);
849
850 if (sceneModeOverrides.count > 0) {
851 // sceneModeOverrides is defined to have 3 entries for each scene mode,
852 // which are AE, AWB, and AF override modes the HAL wants for that scene
853 // mode.
854 const size_t kModesPerSceneMode = 3;
855 if (sceneModeOverrides.count !=
856 availableSceneModes.count * kModesPerSceneMode) {
857 ALOGE("%s: Camera %d: Scene mode override list is an "
858 "unexpected size: %d (expected %d)", __FUNCTION__,
859 cameraId, sceneModeOverrides.count,
860 availableSceneModes.count);
861 return NO_INIT;
862 }
863 for (size_t i = 0; i < availableSceneModes.count; i++) {
864 DeviceInfo::OverrideModes modes;
865 uint8_t aeMode =
866 sceneModeOverrides.data.u8[i * kModesPerSceneMode + 0];
867 switch(aeMode) {
868 case ANDROID_CONTROL_AE_ON:
869 modes.flashMode = FLASH_MODE_OFF;
870 break;
871 case ANDROID_CONTROL_AE_ON_AUTO_FLASH:
872 modes.flashMode = FLASH_MODE_AUTO;
873 break;
874 case ANDROID_CONTROL_AE_ON_ALWAYS_FLASH:
875 modes.flashMode = FLASH_MODE_ON;
876 break;
877 case ANDROID_CONTROL_AE_ON_AUTO_FLASH_REDEYE:
878 modes.flashMode = FLASH_MODE_RED_EYE;
879 break;
880 default:
881 ALOGE("%s: Unknown override AE mode: %d", __FUNCTION__,
882 aeMode);
883 modes.flashMode = FLASH_MODE_INVALID;
884 break;
885 }
886 modes.wbMode =
887 sceneModeOverrides.data.u8[i * kModesPerSceneMode + 1];
888 uint8_t afMode =
889 sceneModeOverrides.data.u8[i * kModesPerSceneMode + 2];
890 switch(afMode) {
891 case ANDROID_CONTROL_AF_OFF:
892 modes.focusMode = fixedLens ?
893 FOCUS_MODE_FIXED : FOCUS_MODE_INFINITY;
894 break;
895 case ANDROID_CONTROL_AF_AUTO:
896 case ANDROID_CONTROL_AF_MACRO:
897 case ANDROID_CONTROL_AF_CONTINUOUS_VIDEO:
898 case ANDROID_CONTROL_AF_CONTINUOUS_PICTURE:
899 case ANDROID_CONTROL_AF_EDOF:
900 modes.focusMode = static_cast<focusMode_t>(afMode);
901 break;
902 default:
903 ALOGE("%s: Unknown override AF mode: %d", __FUNCTION__,
904 afMode);
905 modes.focusMode = FOCUS_MODE_INVALID;
906 break;
907 }
908 fastInfo.sceneModeOverrides.add(availableSceneModes.data.u8[i],
909 modes);
910 }
911 }
912
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700913 fastInfo.arrayWidth = arrayWidth;
914 fastInfo.arrayHeight = arrayHeight;
915 fastInfo.bestFaceDetectMode = bestFaceDetectMode;
916 fastInfo.maxFaces = maxFaces;
917 return OK;
918}
919
Eino-Ville Talvalae382ee22012-10-02 18:14:49 -0700920status_t Parameters::buildQuirks() {
921 camera_metadata_ro_entry_t entry;
922 entry = info->find(ANDROID_QUIRKS_TRIGGER_AF_WITH_AUTO);
923 quirks.triggerAfWithAuto = (entry.count != 0 && entry.data.u8[0] == 1);
924 ALOGV_IF(quirks.triggerAfWithAuto, "Camera %d: Quirk triggerAfWithAuto enabled",
925 cameraId);
926
927 entry = info->find(ANDROID_QUIRKS_USE_ZSL_FORMAT);
928 quirks.useZslFormat = (entry.count != 0 && entry.data.u8[0] == 1);
929 ALOGV_IF(quirks.useZslFormat, "Camera %d: Quirk useZslFormat enabled",
930 cameraId);
931
Igor Murashkin7373cbe2012-09-28 15:30:03 -0700932 entry = info->find(ANDROID_QUIRKS_METERING_CROP_REGION);
933 quirks.meteringCropRegion = (entry.count != 0 && entry.data.u8[0] == 1);
934 ALOGV_IF(quirks.meteringCropRegion, "Camera %d: Quirk meteringCropRegion"
935 " enabled", cameraId);
936
Eino-Ville Talvalae382ee22012-10-02 18:14:49 -0700937 return OK;
938}
939
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700940camera_metadata_ro_entry_t Parameters::staticInfo(uint32_t tag,
941 size_t minCount, size_t maxCount) const {
942 status_t res;
943 camera_metadata_ro_entry_t entry = info->find(tag);
944
945 if (CC_UNLIKELY( entry.count == 0 )) {
946 const char* tagSection = get_camera_metadata_section_name(tag);
947 if (tagSection == NULL) tagSection = "<unknown>";
948 const char* tagName = get_camera_metadata_tag_name(tag);
949 if (tagName == NULL) tagName = "<unknown>";
950
951 ALOGE("Error finding static metadata entry '%s.%s' (%x)",
952 tagSection, tagName, tag);
953 } else if (CC_UNLIKELY(
954 (minCount != 0 && entry.count < minCount) ||
955 (maxCount != 0 && entry.count > maxCount) ) ) {
956 const char* tagSection = get_camera_metadata_section_name(tag);
957 if (tagSection == NULL) tagSection = "<unknown>";
958 const char* tagName = get_camera_metadata_tag_name(tag);
959 if (tagName == NULL) tagName = "<unknown>";
960 ALOGE("Malformed static metadata entry '%s.%s' (%x):"
961 "Expected between %d and %d values, but got %d values",
962 tagSection, tagName, tag, minCount, maxCount, entry.count);
963 }
964
965 return entry;
966}
967
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -0700968status_t Parameters::set(const String8& paramString) {
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700969 status_t res;
970
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -0700971 CameraParameters newParams(paramString);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -0700972
973 // TODO: Currently ignoring any changes to supposedly read-only parameters
974 // such as supported preview sizes, etc. Should probably produce an error if
975 // they're changed.
976
977 /** Extract and verify new parameters */
978
979 size_t i;
980
981 Parameters validatedParams(*this);
982
983 // PREVIEW_SIZE
984 newParams.getPreviewSize(&validatedParams.previewWidth,
985 &validatedParams.previewHeight);
986
987 if (validatedParams.previewWidth != previewWidth ||
988 validatedParams.previewHeight != previewHeight) {
989 if (state >= PREVIEW) {
990 ALOGE("%s: Preview size cannot be updated when preview "
991 "is active! (Currently %d x %d, requested %d x %d",
992 __FUNCTION__,
993 previewWidth, previewHeight,
994 validatedParams.previewWidth, validatedParams.previewHeight);
995 return BAD_VALUE;
996 }
997 camera_metadata_ro_entry_t availablePreviewSizes =
998 staticInfo(ANDROID_SCALER_AVAILABLE_PROCESSED_SIZES);
999 for (i = 0; i < availablePreviewSizes.count; i += 2 ) {
1000 if ((availablePreviewSizes.data.i32[i] ==
1001 validatedParams.previewWidth) &&
1002 (availablePreviewSizes.data.i32[i+1] ==
1003 validatedParams.previewHeight)) break;
1004 }
1005 if (i == availablePreviewSizes.count) {
1006 ALOGE("%s: Requested preview size %d x %d is not supported",
1007 __FUNCTION__, validatedParams.previewWidth,
1008 validatedParams.previewHeight);
1009 return BAD_VALUE;
1010 }
1011 }
1012
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001013 // RECORDING_HINT (always supported)
1014 validatedParams.recordingHint = boolFromString(
1015 newParams.get(CameraParameters::KEY_RECORDING_HINT) );
1016 bool recordingHintChanged = validatedParams.recordingHint != recordingHint;
1017 ALOGV_IF(recordingHintChanged, "%s: Recording hint changed to %d",
1018 __FUNCTION__, recordingHintChanged);
1019
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001020 // PREVIEW_FPS_RANGE
1021 bool fpsRangeChanged = false;
1022 newParams.getPreviewFpsRange(&validatedParams.previewFpsRange[0],
1023 &validatedParams.previewFpsRange[1]);
Eino-Ville Talvalac9d7e4d2012-09-27 14:18:13 -07001024 validatedParams.previewFpsRange[0] /= kFpsToApiScale;
1025 validatedParams.previewFpsRange[1] /= kFpsToApiScale;
1026
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001027 if (validatedParams.previewFpsRange[0] != previewFpsRange[0] ||
1028 validatedParams.previewFpsRange[1] != previewFpsRange[1]) {
1029 fpsRangeChanged = true;
1030 camera_metadata_ro_entry_t availablePreviewFpsRanges =
1031 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_TARGET_FPS_RANGES, 2);
1032 for (i = 0; i < availablePreviewFpsRanges.count; i += 2) {
1033 if ((availablePreviewFpsRanges.data.i32[i] ==
1034 validatedParams.previewFpsRange[0]) &&
1035 (availablePreviewFpsRanges.data.i32[i+1] ==
1036 validatedParams.previewFpsRange[1]) ) {
1037 break;
1038 }
1039 }
1040 if (i == availablePreviewFpsRanges.count) {
1041 ALOGE("%s: Requested preview FPS range %d - %d is not supported",
1042 __FUNCTION__, validatedParams.previewFpsRange[0],
1043 validatedParams.previewFpsRange[1]);
1044 return BAD_VALUE;
1045 }
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001046 validatedParams.previewFps =
1047 fpsFromRange(validatedParams.previewFpsRange[0],
1048 validatedParams.previewFpsRange[1]);
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001049 newParams.setPreviewFrameRate(validatedParams.previewFps);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001050 }
1051
1052 // PREVIEW_FORMAT
1053 validatedParams.previewFormat =
1054 formatStringToEnum(newParams.getPreviewFormat());
1055 if (validatedParams.previewFormat != previewFormat) {
1056 if (state >= PREVIEW) {
1057 ALOGE("%s: Preview format cannot be updated when preview "
1058 "is active!", __FUNCTION__);
1059 return BAD_VALUE;
1060 }
1061 camera_metadata_ro_entry_t availableFormats =
1062 staticInfo(ANDROID_SCALER_AVAILABLE_FORMATS);
1063 for (i = 0; i < availableFormats.count; i++) {
1064 if (availableFormats.data.i32[i] == validatedParams.previewFormat)
1065 break;
1066 }
1067 if (i == availableFormats.count) {
1068 ALOGE("%s: Requested preview format %s (0x%x) is not supported",
1069 __FUNCTION__, newParams.getPreviewFormat(),
1070 validatedParams.previewFormat);
1071 return BAD_VALUE;
1072 }
1073 }
1074
1075 // PREVIEW_FRAME_RATE
1076 // Deprecated, only use if the preview fps range is unchanged this time.
1077 // The single-value FPS is the same as the minimum of the range.
1078 if (!fpsRangeChanged) {
Eino-Ville Talvalad0cec0c2012-09-27 18:08:20 -07001079 validatedParams.previewFps = newParams.getPreviewFrameRate();
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001080 if (validatedParams.previewFps != previewFps || recordingHintChanged) {
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001081 camera_metadata_ro_entry_t availableFrameRates =
1082 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_TARGET_FPS_RANGES);
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001083 /**
1084 * If recording hint is set, find the range that encompasses
1085 * previewFps with the largest min index.
1086 *
1087 * If recording hint is not set, find the range with previewFps
1088 * with the smallest min index.
1089 *
1090 * Either way, in case of multiple ranges, break the tie by
1091 * selecting the smaller range.
1092 */
1093 int targetFps = validatedParams.previewFps;
1094 // all ranges which have targetFps
1095 Vector<Range> candidateRanges;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001096 for (i = 0; i < availableFrameRates.count; i+=2) {
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001097 Range r = {
1098 availableFrameRates.data.i32[i],
1099 availableFrameRates.data.i32[i+1]
1100 };
1101
1102 if (r.min <= targetFps && targetFps <= r.max) {
1103 candidateRanges.push(r);
1104 }
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001105 }
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001106 if (candidateRanges.isEmpty()) {
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001107 ALOGE("%s: Requested preview frame rate %d is not supported",
1108 __FUNCTION__, validatedParams.previewFps);
1109 return BAD_VALUE;
1110 }
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001111 // most applicable range with targetFps
1112 Range bestRange = candidateRanges[0];
1113 for (i = 1; i < candidateRanges.size(); ++i) {
1114 Range r = candidateRanges[i];
1115
1116 // Find by largest minIndex in recording mode
1117 if (validatedParams.recordingHint) {
1118 if (r.min > bestRange.min) {
1119 bestRange = r;
1120 }
1121 else if (r.min == bestRange.min && r.max < bestRange.max) {
1122 bestRange = r;
1123 }
1124 }
1125 // Find by smallest minIndex in preview mode
1126 else {
1127 if (r.min < bestRange.min) {
1128 bestRange = r;
1129 }
1130 else if (r.min == bestRange.min && r.max < bestRange.max) {
1131 bestRange = r;
1132 }
1133 }
1134 }
1135
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001136 validatedParams.previewFpsRange[0] =
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001137 bestRange.min;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001138 validatedParams.previewFpsRange[1] =
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001139 bestRange.max;
1140
1141 ALOGV("%s: New preview FPS range: %d, %d, recordingHint = %d",
1142 __FUNCTION__,
1143 validatedParams.previewFpsRange[0],
1144 validatedParams.previewFpsRange[1],
1145 validatedParams.recordingHint);
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001146 }
Eino-Ville Talvala260f2672012-10-03 14:59:29 -07001147 newParams.set(CameraParameters::KEY_PREVIEW_FPS_RANGE,
1148 String8::format("%d,%d",
1149 validatedParams.previewFpsRange[0] * kFpsToApiScale,
1150 validatedParams.previewFpsRange[1] * kFpsToApiScale));
Igor Murashkinaf3d2882012-10-04 14:22:18 -07001151
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001152 }
1153
1154 // PICTURE_SIZE
1155 newParams.getPictureSize(&validatedParams.pictureWidth,
1156 &validatedParams.pictureHeight);
1157 if (validatedParams.pictureWidth == pictureWidth ||
1158 validatedParams.pictureHeight == pictureHeight) {
1159 camera_metadata_ro_entry_t availablePictureSizes =
1160 staticInfo(ANDROID_SCALER_AVAILABLE_JPEG_SIZES);
1161 for (i = 0; i < availablePictureSizes.count; i+=2) {
1162 if ((availablePictureSizes.data.i32[i] ==
1163 validatedParams.pictureWidth) &&
1164 (availablePictureSizes.data.i32[i+1] ==
1165 validatedParams.pictureHeight)) break;
1166 }
1167 if (i == availablePictureSizes.count) {
1168 ALOGE("%s: Requested picture size %d x %d is not supported",
1169 __FUNCTION__, validatedParams.pictureWidth,
1170 validatedParams.pictureHeight);
1171 return BAD_VALUE;
1172 }
1173 }
1174
1175 // JPEG_THUMBNAIL_WIDTH/HEIGHT
1176 validatedParams.jpegThumbSize[0] =
1177 newParams.getInt(CameraParameters::KEY_JPEG_THUMBNAIL_WIDTH);
1178 validatedParams.jpegThumbSize[1] =
1179 newParams.getInt(CameraParameters::KEY_JPEG_THUMBNAIL_HEIGHT);
1180 if (validatedParams.jpegThumbSize[0] != jpegThumbSize[0] ||
1181 validatedParams.jpegThumbSize[1] != jpegThumbSize[1]) {
1182 camera_metadata_ro_entry_t availableJpegThumbSizes =
1183 staticInfo(ANDROID_JPEG_AVAILABLE_THUMBNAIL_SIZES);
1184 for (i = 0; i < availableJpegThumbSizes.count; i+=2) {
1185 if ((availableJpegThumbSizes.data.i32[i] ==
1186 validatedParams.jpegThumbSize[0]) &&
1187 (availableJpegThumbSizes.data.i32[i+1] ==
1188 validatedParams.jpegThumbSize[1])) break;
1189 }
1190 if (i == availableJpegThumbSizes.count) {
1191 ALOGE("%s: Requested JPEG thumbnail size %d x %d is not supported",
1192 __FUNCTION__, validatedParams.jpegThumbSize[0],
1193 validatedParams.jpegThumbSize[1]);
1194 return BAD_VALUE;
1195 }
1196 }
1197
1198 // JPEG_THUMBNAIL_QUALITY
1199 validatedParams.jpegThumbQuality =
1200 newParams.getInt(CameraParameters::KEY_JPEG_THUMBNAIL_QUALITY);
1201 if (validatedParams.jpegThumbQuality < 0 ||
1202 validatedParams.jpegThumbQuality > 100) {
1203 ALOGE("%s: Requested JPEG thumbnail quality %d is not supported",
1204 __FUNCTION__, validatedParams.jpegThumbQuality);
1205 return BAD_VALUE;
1206 }
1207
1208 // JPEG_QUALITY
1209 validatedParams.jpegQuality =
1210 newParams.getInt(CameraParameters::KEY_JPEG_QUALITY);
1211 if (validatedParams.jpegQuality < 0 || validatedParams.jpegQuality > 100) {
1212 ALOGE("%s: Requested JPEG quality %d is not supported",
1213 __FUNCTION__, validatedParams.jpegQuality);
1214 return BAD_VALUE;
1215 }
1216
1217 // ROTATION
1218 validatedParams.jpegRotation =
1219 newParams.getInt(CameraParameters::KEY_ROTATION);
1220 if (validatedParams.jpegRotation != 0 &&
1221 validatedParams.jpegRotation != 90 &&
1222 validatedParams.jpegRotation != 180 &&
1223 validatedParams.jpegRotation != 270) {
1224 ALOGE("%s: Requested picture rotation angle %d is not supported",
1225 __FUNCTION__, validatedParams.jpegRotation);
1226 return BAD_VALUE;
1227 }
1228
1229 // GPS
1230
1231 const char *gpsLatStr =
1232 newParams.get(CameraParameters::KEY_GPS_LATITUDE);
1233 if (gpsLatStr != NULL) {
1234 const char *gpsLongStr =
1235 newParams.get(CameraParameters::KEY_GPS_LONGITUDE);
1236 const char *gpsAltitudeStr =
1237 newParams.get(CameraParameters::KEY_GPS_ALTITUDE);
1238 const char *gpsTimeStr =
1239 newParams.get(CameraParameters::KEY_GPS_TIMESTAMP);
1240 const char *gpsProcMethodStr =
1241 newParams.get(CameraParameters::KEY_GPS_PROCESSING_METHOD);
1242 if (gpsLongStr == NULL ||
1243 gpsAltitudeStr == NULL ||
1244 gpsTimeStr == NULL ||
1245 gpsProcMethodStr == NULL) {
1246 ALOGE("%s: Incomplete set of GPS parameters provided",
1247 __FUNCTION__);
1248 return BAD_VALUE;
1249 }
1250 char *endPtr;
1251 errno = 0;
1252 validatedParams.gpsCoordinates[0] = strtod(gpsLatStr, &endPtr);
1253 if (errno || endPtr == gpsLatStr) {
1254 ALOGE("%s: Malformed GPS latitude: %s", __FUNCTION__, gpsLatStr);
1255 return BAD_VALUE;
1256 }
1257 errno = 0;
1258 validatedParams.gpsCoordinates[1] = strtod(gpsLongStr, &endPtr);
1259 if (errno || endPtr == gpsLongStr) {
1260 ALOGE("%s: Malformed GPS longitude: %s", __FUNCTION__, gpsLongStr);
1261 return BAD_VALUE;
1262 }
1263 errno = 0;
1264 validatedParams.gpsCoordinates[2] = strtod(gpsAltitudeStr, &endPtr);
1265 if (errno || endPtr == gpsAltitudeStr) {
1266 ALOGE("%s: Malformed GPS altitude: %s", __FUNCTION__,
1267 gpsAltitudeStr);
1268 return BAD_VALUE;
1269 }
1270 errno = 0;
1271 validatedParams.gpsTimestamp = strtoll(gpsTimeStr, &endPtr, 10);
1272 if (errno || endPtr == gpsTimeStr) {
1273 ALOGE("%s: Malformed GPS timestamp: %s", __FUNCTION__, gpsTimeStr);
1274 return BAD_VALUE;
1275 }
1276 validatedParams.gpsProcessingMethod = gpsProcMethodStr;
1277
1278 validatedParams.gpsEnabled = true;
1279 } else {
1280 validatedParams.gpsEnabled = false;
1281 }
1282
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001283 // EFFECT
1284 validatedParams.effectMode = effectModeStringToEnum(
1285 newParams.get(CameraParameters::KEY_EFFECT) );
1286 if (validatedParams.effectMode != effectMode) {
1287 camera_metadata_ro_entry_t availableEffectModes =
1288 staticInfo(ANDROID_CONTROL_AVAILABLE_EFFECTS);
1289 for (i = 0; i < availableEffectModes.count; i++) {
1290 if (validatedParams.effectMode == availableEffectModes.data.u8[i]) break;
1291 }
1292 if (i == availableEffectModes.count) {
1293 ALOGE("%s: Requested effect mode \"%s\" is not supported",
1294 __FUNCTION__,
1295 newParams.get(CameraParameters::KEY_EFFECT) );
1296 return BAD_VALUE;
1297 }
1298 }
1299
1300 // ANTIBANDING
1301 validatedParams.antibandingMode = abModeStringToEnum(
1302 newParams.get(CameraParameters::KEY_ANTIBANDING) );
1303 if (validatedParams.antibandingMode != antibandingMode) {
1304 camera_metadata_ro_entry_t availableAbModes =
1305 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_ANTIBANDING_MODES);
1306 for (i = 0; i < availableAbModes.count; i++) {
1307 if (validatedParams.antibandingMode == availableAbModes.data.u8[i])
1308 break;
1309 }
1310 if (i == availableAbModes.count) {
1311 ALOGE("%s: Requested antibanding mode \"%s\" is not supported",
1312 __FUNCTION__,
1313 newParams.get(CameraParameters::KEY_ANTIBANDING));
1314 return BAD_VALUE;
1315 }
1316 }
1317
1318 // SCENE_MODE
1319 validatedParams.sceneMode = sceneModeStringToEnum(
1320 newParams.get(CameraParameters::KEY_SCENE_MODE) );
1321 if (validatedParams.sceneMode != sceneMode &&
1322 validatedParams.sceneMode !=
1323 ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED) {
1324 camera_metadata_ro_entry_t availableSceneModes =
1325 staticInfo(ANDROID_CONTROL_AVAILABLE_SCENE_MODES);
1326 for (i = 0; i < availableSceneModes.count; i++) {
1327 if (validatedParams.sceneMode == availableSceneModes.data.u8[i])
1328 break;
1329 }
1330 if (i == availableSceneModes.count) {
1331 ALOGE("%s: Requested scene mode \"%s\" is not supported",
1332 __FUNCTION__,
1333 newParams.get(CameraParameters::KEY_SCENE_MODE));
1334 return BAD_VALUE;
1335 }
1336 }
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001337 bool sceneModeSet =
1338 validatedParams.sceneMode != ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001339
1340 // FLASH_MODE
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001341 if (sceneModeSet) {
1342 validatedParams.flashMode =
1343 fastInfo.sceneModeOverrides.
1344 valueFor(validatedParams.sceneMode).flashMode;
1345 } else {
1346 validatedParams.flashMode = FLASH_MODE_INVALID;
1347 }
1348 if (validatedParams.flashMode == FLASH_MODE_INVALID) {
1349 validatedParams.flashMode = flashModeStringToEnum(
1350 newParams.get(CameraParameters::KEY_FLASH_MODE) );
1351 }
1352
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001353 if (validatedParams.flashMode != flashMode) {
1354 camera_metadata_ro_entry_t flashAvailable =
1355 staticInfo(ANDROID_FLASH_AVAILABLE, 1, 1);
1356 if (!flashAvailable.data.u8[0] &&
1357 validatedParams.flashMode != Parameters::FLASH_MODE_OFF) {
1358 ALOGE("%s: Requested flash mode \"%s\" is not supported: "
1359 "No flash on device", __FUNCTION__,
1360 newParams.get(CameraParameters::KEY_FLASH_MODE));
1361 return BAD_VALUE;
1362 } else if (validatedParams.flashMode == Parameters::FLASH_MODE_RED_EYE) {
1363 camera_metadata_ro_entry_t availableAeModes =
1364 staticInfo(ANDROID_CONTROL_AE_AVAILABLE_MODES);
1365 for (i = 0; i < availableAeModes.count; i++) {
1366 if (validatedParams.flashMode == availableAeModes.data.u8[i])
1367 break;
1368 }
1369 if (i == availableAeModes.count) {
1370 ALOGE("%s: Requested flash mode \"%s\" is not supported",
1371 __FUNCTION__,
1372 newParams.get(CameraParameters::KEY_FLASH_MODE));
1373 return BAD_VALUE;
1374 }
1375 } else if (validatedParams.flashMode == -1) {
1376 ALOGE("%s: Requested flash mode \"%s\" is unknown",
1377 __FUNCTION__,
1378 newParams.get(CameraParameters::KEY_FLASH_MODE));
1379 return BAD_VALUE;
1380 }
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001381 // Update in case of override
1382 newParams.set(CameraParameters::KEY_FLASH_MODE,
1383 flashModeEnumToString(validatedParams.flashMode));
1384 }
1385
1386 // WHITE_BALANCE
1387 if (sceneModeSet) {
1388 validatedParams.wbMode =
1389 fastInfo.sceneModeOverrides.
1390 valueFor(validatedParams.sceneMode).wbMode;
1391 } else {
1392 validatedParams.wbMode = ANDROID_CONTROL_AWB_OFF;
1393 }
1394 if (validatedParams.wbMode == ANDROID_CONTROL_AWB_OFF) {
1395 validatedParams.wbMode = wbModeStringToEnum(
1396 newParams.get(CameraParameters::KEY_WHITE_BALANCE) );
1397 }
1398 if (validatedParams.wbMode != wbMode) {
1399 camera_metadata_ro_entry_t availableWbModes =
1400 staticInfo(ANDROID_CONTROL_AWB_AVAILABLE_MODES);
1401 for (i = 0; i < availableWbModes.count; i++) {
1402 if (validatedParams.wbMode == availableWbModes.data.u8[i]) break;
1403 }
1404 if (i == availableWbModes.count) {
1405 ALOGE("%s: Requested white balance mode %s is not supported",
1406 __FUNCTION__,
1407 newParams.get(CameraParameters::KEY_WHITE_BALANCE));
1408 return BAD_VALUE;
1409 }
1410 // Update in case of override
1411 newParams.set(CameraParameters::KEY_WHITE_BALANCE,
1412 wbModeEnumToString(validatedParams.wbMode));
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001413 }
1414
1415 // FOCUS_MODE
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001416 if (sceneModeSet) {
1417 validatedParams.focusMode =
1418 fastInfo.sceneModeOverrides.
1419 valueFor(validatedParams.sceneMode).focusMode;
1420 } else {
1421 validatedParams.focusMode = FOCUS_MODE_INVALID;
1422 }
1423 if (validatedParams.focusMode == FOCUS_MODE_INVALID) {
1424 validatedParams.focusMode = focusModeStringToEnum(
1425 newParams.get(CameraParameters::KEY_FOCUS_MODE) );
1426 }
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001427 if (validatedParams.focusMode != focusMode) {
1428 validatedParams.currentAfTriggerId = -1;
1429 if (validatedParams.focusMode != Parameters::FOCUS_MODE_FIXED) {
1430 camera_metadata_ro_entry_t minFocusDistance =
1431 staticInfo(ANDROID_LENS_MINIMUM_FOCUS_DISTANCE);
1432 if (minFocusDistance.data.f[0] == 0) {
1433 ALOGE("%s: Requested focus mode \"%s\" is not available: "
1434 "fixed focus lens",
1435 __FUNCTION__,
1436 newParams.get(CameraParameters::KEY_FOCUS_MODE));
1437 return BAD_VALUE;
1438 } else if (validatedParams.focusMode !=
1439 Parameters::FOCUS_MODE_INFINITY) {
1440 camera_metadata_ro_entry_t availableFocusModes =
1441 staticInfo(ANDROID_CONTROL_AF_AVAILABLE_MODES);
1442 for (i = 0; i < availableFocusModes.count; i++) {
1443 if (validatedParams.focusMode ==
1444 availableFocusModes.data.u8[i]) break;
1445 }
1446 if (i == availableFocusModes.count) {
1447 ALOGE("%s: Requested focus mode \"%s\" is not supported",
1448 __FUNCTION__,
1449 newParams.get(CameraParameters::KEY_FOCUS_MODE));
1450 return BAD_VALUE;
1451 }
1452 }
1453 }
Eino-Ville Talvala95069fe2012-10-04 00:56:40 -07001454 // Always reset shadow focus mode to avoid reverting settings
1455 shadowFocusMode = FOCUS_MODE_INVALID;
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001456 // Update in case of override
1457 newParams.set(CameraParameters::KEY_FOCUS_MODE,
1458 focusModeEnumToString(validatedParams.focusMode));
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001459 } else {
1460 validatedParams.currentAfTriggerId = currentAfTriggerId;
1461 }
1462
1463 // FOCUS_AREAS
1464 res = parseAreas(newParams.get(CameraParameters::KEY_FOCUS_AREAS),
1465 &validatedParams.focusingAreas);
1466 size_t max3aRegions =
1467 (size_t)staticInfo(ANDROID_CONTROL_MAX_REGIONS, 1, 1).data.i32[0];
1468 if (res == OK) res = validateAreas(validatedParams.focusingAreas,
1469 max3aRegions);
1470 if (res != OK) {
1471 ALOGE("%s: Requested focus areas are malformed: %s",
1472 __FUNCTION__, newParams.get(CameraParameters::KEY_FOCUS_AREAS));
1473 return BAD_VALUE;
1474 }
1475
1476 // EXPOSURE_COMPENSATION
1477 validatedParams.exposureCompensation =
1478 newParams.getInt(CameraParameters::KEY_EXPOSURE_COMPENSATION);
1479 camera_metadata_ro_entry_t exposureCompensationRange =
1480 staticInfo(ANDROID_CONTROL_AE_EXP_COMPENSATION_RANGE);
1481 if ((validatedParams.exposureCompensation <
1482 exposureCompensationRange.data.i32[0]) ||
1483 (validatedParams.exposureCompensation >
1484 exposureCompensationRange.data.i32[1])) {
1485 ALOGE("%s: Requested exposure compensation index is out of bounds: %d",
1486 __FUNCTION__, validatedParams.exposureCompensation);
1487 return BAD_VALUE;
1488 }
1489
1490 // AUTO_EXPOSURE_LOCK (always supported)
1491 validatedParams.autoExposureLock = boolFromString(
1492 newParams.get(CameraParameters::KEY_AUTO_EXPOSURE_LOCK));
1493
1494 // AUTO_WHITEBALANCE_LOCK (always supported)
1495 validatedParams.autoWhiteBalanceLock = boolFromString(
1496 newParams.get(CameraParameters::KEY_AUTO_WHITEBALANCE_LOCK));
1497
1498 // METERING_AREAS
1499 res = parseAreas(newParams.get(CameraParameters::KEY_METERING_AREAS),
1500 &validatedParams.meteringAreas);
1501 if (res == OK) {
1502 res = validateAreas(validatedParams.meteringAreas, max3aRegions);
1503 }
1504 if (res != OK) {
1505 ALOGE("%s: Requested metering areas are malformed: %s",
1506 __FUNCTION__,
1507 newParams.get(CameraParameters::KEY_METERING_AREAS));
1508 return BAD_VALUE;
1509 }
1510
1511 // ZOOM
1512 validatedParams.zoom = newParams.getInt(CameraParameters::KEY_ZOOM);
1513 if (validatedParams.zoom < 0 || validatedParams.zoom > (int)NUM_ZOOM_STEPS) {
1514 ALOGE("%s: Requested zoom level %d is not supported",
1515 __FUNCTION__, validatedParams.zoom);
1516 return BAD_VALUE;
1517 }
1518
1519 // VIDEO_SIZE
1520 newParams.getVideoSize(&validatedParams.videoWidth,
1521 &validatedParams.videoHeight);
1522 if (validatedParams.videoWidth != videoWidth ||
1523 validatedParams.videoHeight != videoHeight) {
1524 if (state == RECORD) {
1525 ALOGE("%s: Video size cannot be updated when recording is active!",
1526 __FUNCTION__);
1527 return BAD_VALUE;
1528 }
1529 camera_metadata_ro_entry_t availableVideoSizes =
1530 staticInfo(ANDROID_SCALER_AVAILABLE_PROCESSED_SIZES);
1531 for (i = 0; i < availableVideoSizes.count; i += 2 ) {
1532 if ((availableVideoSizes.data.i32[i] ==
1533 validatedParams.videoWidth) &&
1534 (availableVideoSizes.data.i32[i+1] ==
1535 validatedParams.videoHeight)) break;
1536 }
1537 if (i == availableVideoSizes.count) {
1538 ALOGE("%s: Requested video size %d x %d is not supported",
1539 __FUNCTION__, validatedParams.videoWidth,
1540 validatedParams.videoHeight);
1541 return BAD_VALUE;
1542 }
1543 }
1544
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001545 // VIDEO_STABILIZATION
1546 validatedParams.videoStabilization = boolFromString(
1547 newParams.get(CameraParameters::KEY_VIDEO_STABILIZATION) );
1548 camera_metadata_ro_entry_t availableVideoStabilizationModes =
1549 staticInfo(ANDROID_CONTROL_AVAILABLE_VIDEO_STABILIZATION_MODES);
1550 if (validatedParams.videoStabilization &&
1551 availableVideoStabilizationModes.count == 1) {
1552 ALOGE("%s: Video stabilization not supported", __FUNCTION__);
1553 }
1554
1555 /** Update internal parameters */
1556
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001557 *this = validatedParams;
1558
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001559 // Need to flatten again in case of overrides
1560 paramsFlattened = newParams.flatten();
1561 params = newParams;
1562
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001563 return OK;
1564}
1565
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001566status_t Parameters::updateRequest(CameraMetadata *request) const {
1567 ATRACE_CALL();
1568 status_t res;
1569
1570 uint8_t metadataMode = ANDROID_REQUEST_METADATA_FULL;
1571 res = request->update(ANDROID_REQUEST_METADATA_MODE,
1572 &metadataMode, 1);
1573 if (res != OK) return res;
1574
1575 res = request->update(ANDROID_CONTROL_AE_TARGET_FPS_RANGE,
1576 previewFpsRange, 2);
1577 if (res != OK) return res;
1578
Eino-Ville Talvala96150cf2012-09-03 10:03:26 -07001579 uint8_t reqWbLock = autoWhiteBalanceLock ?
1580 ANDROID_CONTROL_AWB_LOCK_ON : ANDROID_CONTROL_AWB_LOCK_OFF;
1581 res = request->update(ANDROID_CONTROL_AWB_LOCK,
1582 &reqWbLock, 1);
1583
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001584 res = request->update(ANDROID_CONTROL_EFFECT_MODE,
1585 &effectMode, 1);
1586 if (res != OK) return res;
1587 res = request->update(ANDROID_CONTROL_AE_ANTIBANDING_MODE,
1588 &antibandingMode, 1);
1589 if (res != OK) return res;
1590
Eino-Ville Talvala61c0cc62012-09-07 14:26:29 -07001591 // android.hardware.Camera requires that when face detect is enabled, the
1592 // camera is in a face-priority mode. HAL2 splits this into separate parts
1593 // (face detection statistics and face priority scene mode). Map from other
1594 // to the other.
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001595 bool sceneModeActive =
1596 sceneMode != (uint8_t)ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED;
Eino-Ville Talvala61c0cc62012-09-07 14:26:29 -07001597 uint8_t reqControlMode = ANDROID_CONTROL_AUTO;
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001598 if (enableFaceDetect || sceneModeActive) {
Eino-Ville Talvala61c0cc62012-09-07 14:26:29 -07001599 reqControlMode = ANDROID_CONTROL_USE_SCENE_MODE;
1600 }
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001601 res = request->update(ANDROID_CONTROL_MODE,
1602 &reqControlMode, 1);
1603 if (res != OK) return res;
Eino-Ville Talvala61c0cc62012-09-07 14:26:29 -07001604
Eino-Ville Talvalad14de472012-09-17 16:43:14 -07001605 uint8_t reqSceneMode =
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001606 sceneModeActive ? sceneMode :
Eino-Ville Talvalad14de472012-09-17 16:43:14 -07001607 enableFaceDetect ? (uint8_t)ANDROID_CONTROL_SCENE_MODE_FACE_PRIORITY :
1608 (uint8_t)ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED;
Eino-Ville Talvala61c0cc62012-09-07 14:26:29 -07001609 res = request->update(ANDROID_CONTROL_SCENE_MODE,
1610 &reqSceneMode, 1);
1611 if (res != OK) return res;
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001612
1613 uint8_t reqFlashMode = ANDROID_FLASH_OFF;
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001614 uint8_t reqAeMode = ANDROID_CONTROL_AE_OFF;
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001615 switch (flashMode) {
1616 case Parameters::FLASH_MODE_OFF:
1617 reqAeMode = ANDROID_CONTROL_AE_ON; break;
1618 case Parameters::FLASH_MODE_AUTO:
1619 reqAeMode = ANDROID_CONTROL_AE_ON_AUTO_FLASH; break;
1620 case Parameters::FLASH_MODE_ON:
1621 reqAeMode = ANDROID_CONTROL_AE_ON_ALWAYS_FLASH; break;
1622 case Parameters::FLASH_MODE_TORCH:
1623 reqAeMode = ANDROID_CONTROL_AE_ON;
1624 reqFlashMode = ANDROID_FLASH_TORCH;
1625 break;
1626 case Parameters::FLASH_MODE_RED_EYE:
1627 reqAeMode = ANDROID_CONTROL_AE_ON_AUTO_FLASH_REDEYE; break;
1628 default:
1629 ALOGE("%s: Camera %d: Unknown flash mode %d", __FUNCTION__,
1630 cameraId, flashMode);
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001631 return BAD_VALUE;
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001632 }
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001633 res = request->update(ANDROID_FLASH_MODE,
1634 &reqFlashMode, 1);
1635 if (res != OK) return res;
1636 res = request->update(ANDROID_CONTROL_AE_MODE,
1637 &reqAeMode, 1);
1638 if (res != OK) return res;
1639
Eino-Ville Talvala96150cf2012-09-03 10:03:26 -07001640 uint8_t reqAeLock = autoExposureLock ?
1641 ANDROID_CONTROL_AE_LOCK_ON : ANDROID_CONTROL_AE_LOCK_OFF;
1642 res = request->update(ANDROID_CONTROL_AE_LOCK,
1643 &reqAeLock, 1);
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001644 if (res != OK) return res;
1645
1646 res = request->update(ANDROID_CONTROL_AWB_MODE,
1647 &wbMode, 1);
1648 if (res != OK) return res;
Eino-Ville Talvala96150cf2012-09-03 10:03:26 -07001649
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001650 float reqFocusDistance = 0; // infinity focus in diopters
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001651 uint8_t reqFocusMode = ANDROID_CONTROL_AF_OFF;
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001652 switch (focusMode) {
1653 case Parameters::FOCUS_MODE_AUTO:
1654 case Parameters::FOCUS_MODE_MACRO:
1655 case Parameters::FOCUS_MODE_CONTINUOUS_VIDEO:
1656 case Parameters::FOCUS_MODE_CONTINUOUS_PICTURE:
1657 case Parameters::FOCUS_MODE_EDOF:
1658 reqFocusMode = focusMode;
1659 break;
1660 case Parameters::FOCUS_MODE_INFINITY:
1661 case Parameters::FOCUS_MODE_FIXED:
1662 reqFocusMode = ANDROID_CONTROL_AF_OFF;
1663 break;
1664 default:
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001665 ALOGE("%s: Camera %d: Unknown focus mode %d", __FUNCTION__,
1666 cameraId, focusMode);
1667 return BAD_VALUE;
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001668 }
1669 res = request->update(ANDROID_LENS_FOCUS_DISTANCE,
1670 &reqFocusDistance, 1);
1671 if (res != OK) return res;
1672 res = request->update(ANDROID_CONTROL_AF_MODE,
1673 &reqFocusMode, 1);
1674 if (res != OK) return res;
1675
1676 size_t reqFocusingAreasSize = focusingAreas.size() * 5;
1677 int32_t *reqFocusingAreas = new int32_t[reqFocusingAreasSize];
1678 for (size_t i = 0; i < reqFocusingAreasSize; i += 5) {
1679 if (focusingAreas[i].weight != 0) {
1680 reqFocusingAreas[i + 0] =
1681 normalizedXToArray(focusingAreas[i].left);
1682 reqFocusingAreas[i + 1] =
1683 normalizedYToArray(focusingAreas[i].top);
1684 reqFocusingAreas[i + 2] =
1685 normalizedXToArray(focusingAreas[i].right);
1686 reqFocusingAreas[i + 3] =
1687 normalizedYToArray(focusingAreas[i].bottom);
1688 } else {
1689 reqFocusingAreas[i + 0] = 0;
1690 reqFocusingAreas[i + 1] = 0;
1691 reqFocusingAreas[i + 2] = 0;
1692 reqFocusingAreas[i + 3] = 0;
1693 }
1694 reqFocusingAreas[i + 4] = focusingAreas[i].weight;
1695 }
1696 res = request->update(ANDROID_CONTROL_AF_REGIONS,
1697 reqFocusingAreas, reqFocusingAreasSize);
1698 if (res != OK) return res;
1699 delete[] reqFocusingAreas;
1700
1701 res = request->update(ANDROID_CONTROL_AE_EXP_COMPENSATION,
1702 &exposureCompensation, 1);
1703 if (res != OK) return res;
1704
1705 size_t reqMeteringAreasSize = meteringAreas.size() * 5;
1706 int32_t *reqMeteringAreas = new int32_t[reqMeteringAreasSize];
1707 for (size_t i = 0; i < reqMeteringAreasSize; i += 5) {
1708 if (meteringAreas[i].weight != 0) {
1709 reqMeteringAreas[i + 0] =
1710 normalizedXToArray(meteringAreas[i].left);
1711 reqMeteringAreas[i + 1] =
1712 normalizedYToArray(meteringAreas[i].top);
1713 reqMeteringAreas[i + 2] =
1714 normalizedXToArray(meteringAreas[i].right);
1715 reqMeteringAreas[i + 3] =
1716 normalizedYToArray(meteringAreas[i].bottom);
1717 } else {
1718 reqMeteringAreas[i + 0] = 0;
1719 reqMeteringAreas[i + 1] = 0;
1720 reqMeteringAreas[i + 2] = 0;
1721 reqMeteringAreas[i + 3] = 0;
1722 }
1723 reqMeteringAreas[i + 4] = meteringAreas[i].weight;
1724 }
1725 res = request->update(ANDROID_CONTROL_AE_REGIONS,
1726 reqMeteringAreas, reqMeteringAreasSize);
1727 if (res != OK) return res;
1728
1729 res = request->update(ANDROID_CONTROL_AWB_REGIONS,
1730 reqMeteringAreas, reqMeteringAreasSize);
1731 if (res != OK) return res;
1732 delete[] reqMeteringAreas;
1733
Igor Murashkin7373cbe2012-09-28 15:30:03 -07001734 /* don't include jpeg thumbnail size - it's valid for
1735 it to be set to (0,0), meaning 'no thumbnail' */
1736 CropRegion crop = calculateCropRegion( (CropRegion::Outputs)(
1737 CropRegion::OUTPUT_PREVIEW |
1738 CropRegion::OUTPUT_VIDEO |
1739 CropRegion::OUTPUT_PICTURE ));
Igor Murashkin018d2282012-09-18 18:23:49 -07001740 int32_t reqCropRegion[3] = { crop.left, crop.top, crop.width };
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001741 res = request->update(ANDROID_SCALER_CROP_REGION,
1742 reqCropRegion, 3);
1743 if (res != OK) return res;
1744
Eino-Ville Talvalada6665c2012-08-29 17:37:16 -07001745 uint8_t reqVstabMode = videoStabilization ?
1746 ANDROID_CONTROL_VIDEO_STABILIZATION_ON :
1747 ANDROID_CONTROL_VIDEO_STABILIZATION_OFF;
1748 res = request->update(ANDROID_CONTROL_VIDEO_STABILIZATION_MODE,
1749 &reqVstabMode, 1);
1750 if (res != OK) return res;
1751
1752 uint8_t reqFaceDetectMode = enableFaceDetect ?
1753 fastInfo.bestFaceDetectMode :
1754 (uint8_t)ANDROID_STATS_FACE_DETECTION_OFF;
1755 res = request->update(ANDROID_STATS_FACE_DETECT_MODE,
1756 &reqFaceDetectMode, 1);
1757 if (res != OK) return res;
1758
1759 return OK;
1760}
1761
Eino-Ville Talvalaec771082012-10-04 13:21:08 -07001762status_t Parameters::updateRequestJpeg(CameraMetadata *request) const {
1763 status_t res;
1764
1765 res = request->update(ANDROID_JPEG_THUMBNAIL_SIZE,
1766 jpegThumbSize, 2);
1767 if (res != OK) return res;
1768 res = request->update(ANDROID_JPEG_THUMBNAIL_QUALITY,
1769 &jpegThumbQuality, 1);
1770 if (res != OK) return res;
1771 res = request->update(ANDROID_JPEG_QUALITY,
1772 &jpegQuality, 1);
1773 if (res != OK) return res;
1774 res = request->update(
1775 ANDROID_JPEG_ORIENTATION,
1776 &jpegRotation, 1);
1777 if (res != OK) return res;
1778
1779 if (gpsEnabled) {
1780 res = request->update(
1781 ANDROID_JPEG_GPS_COORDINATES,
1782 gpsCoordinates, 3);
1783 if (res != OK) return res;
1784 res = request->update(
1785 ANDROID_JPEG_GPS_TIMESTAMP,
1786 &gpsTimestamp, 1);
1787 if (res != OK) return res;
1788 res = request->update(
1789 ANDROID_JPEG_GPS_PROCESSING_METHOD,
1790 gpsProcessingMethod);
1791 if (res != OK) return res;
1792 } else {
1793 res = request->erase(ANDROID_JPEG_GPS_COORDINATES);
1794 if (res != OK) return res;
1795 res = request->erase(ANDROID_JPEG_GPS_TIMESTAMP);
1796 if (res != OK) return res;
1797 res = request->erase(ANDROID_JPEG_GPS_PROCESSING_METHOD);
1798 if (res != OK) return res;
1799 }
1800 return OK;
1801}
1802
1803
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001804const char* Parameters::getStateName(State state) {
1805#define CASE_ENUM_TO_CHAR(x) case x: return(#x); break;
1806 switch(state) {
1807 CASE_ENUM_TO_CHAR(DISCONNECTED)
1808 CASE_ENUM_TO_CHAR(STOPPED)
1809 CASE_ENUM_TO_CHAR(WAITING_FOR_PREVIEW_WINDOW)
1810 CASE_ENUM_TO_CHAR(PREVIEW)
1811 CASE_ENUM_TO_CHAR(RECORD)
1812 CASE_ENUM_TO_CHAR(STILL_CAPTURE)
1813 CASE_ENUM_TO_CHAR(VIDEO_SNAPSHOT)
1814 default:
1815 return "Unknown state!";
1816 break;
1817 }
1818#undef CASE_ENUM_TO_CHAR
1819}
1820
1821int Parameters::formatStringToEnum(const char *format) {
1822 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07001823 !format ?
1824 HAL_PIXEL_FORMAT_YCrCb_420_SP :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001825 !strcmp(format, CameraParameters::PIXEL_FORMAT_YUV422SP) ?
1826 HAL_PIXEL_FORMAT_YCbCr_422_SP : // NV16
1827 !strcmp(format, CameraParameters::PIXEL_FORMAT_YUV420SP) ?
1828 HAL_PIXEL_FORMAT_YCrCb_420_SP : // NV21
1829 !strcmp(format, CameraParameters::PIXEL_FORMAT_YUV422I) ?
1830 HAL_PIXEL_FORMAT_YCbCr_422_I : // YUY2
1831 !strcmp(format, CameraParameters::PIXEL_FORMAT_YUV420P) ?
1832 HAL_PIXEL_FORMAT_YV12 : // YV12
1833 !strcmp(format, CameraParameters::PIXEL_FORMAT_RGB565) ?
1834 HAL_PIXEL_FORMAT_RGB_565 : // RGB565
1835 !strcmp(format, CameraParameters::PIXEL_FORMAT_RGBA8888) ?
1836 HAL_PIXEL_FORMAT_RGBA_8888 : // RGB8888
1837 !strcmp(format, CameraParameters::PIXEL_FORMAT_BAYER_RGGB) ?
1838 HAL_PIXEL_FORMAT_RAW_SENSOR : // Raw sensor data
1839 -1;
1840}
1841
1842const char* Parameters::formatEnumToString(int format) {
1843 const char *fmt;
1844 switch(format) {
1845 case HAL_PIXEL_FORMAT_YCbCr_422_SP: // NV16
1846 fmt = CameraParameters::PIXEL_FORMAT_YUV422SP;
1847 break;
1848 case HAL_PIXEL_FORMAT_YCrCb_420_SP: // NV21
1849 fmt = CameraParameters::PIXEL_FORMAT_YUV420SP;
1850 break;
1851 case HAL_PIXEL_FORMAT_YCbCr_422_I: // YUY2
1852 fmt = CameraParameters::PIXEL_FORMAT_YUV422I;
1853 break;
1854 case HAL_PIXEL_FORMAT_YV12: // YV12
1855 fmt = CameraParameters::PIXEL_FORMAT_YUV420P;
1856 break;
1857 case HAL_PIXEL_FORMAT_RGB_565: // RGB565
1858 fmt = CameraParameters::PIXEL_FORMAT_RGB565;
1859 break;
1860 case HAL_PIXEL_FORMAT_RGBA_8888: // RGBA8888
1861 fmt = CameraParameters::PIXEL_FORMAT_RGBA8888;
1862 break;
1863 case HAL_PIXEL_FORMAT_RAW_SENSOR:
1864 ALOGW("Raw sensor preview format requested.");
1865 fmt = CameraParameters::PIXEL_FORMAT_BAYER_RGGB;
1866 break;
1867 default:
1868 ALOGE("%s: Unknown preview format: %x",
1869 __FUNCTION__, format);
1870 fmt = NULL;
1871 break;
1872 }
1873 return fmt;
1874}
1875
1876int Parameters::wbModeStringToEnum(const char *wbMode) {
1877 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07001878 !wbMode ?
1879 ANDROID_CONTROL_AWB_AUTO :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001880 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_AUTO) ?
1881 ANDROID_CONTROL_AWB_AUTO :
1882 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_INCANDESCENT) ?
1883 ANDROID_CONTROL_AWB_INCANDESCENT :
1884 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_FLUORESCENT) ?
1885 ANDROID_CONTROL_AWB_FLUORESCENT :
1886 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_WARM_FLUORESCENT) ?
1887 ANDROID_CONTROL_AWB_WARM_FLUORESCENT :
1888 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_DAYLIGHT) ?
1889 ANDROID_CONTROL_AWB_DAYLIGHT :
1890 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_CLOUDY_DAYLIGHT) ?
1891 ANDROID_CONTROL_AWB_CLOUDY_DAYLIGHT :
1892 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_TWILIGHT) ?
1893 ANDROID_CONTROL_AWB_TWILIGHT :
1894 !strcmp(wbMode, CameraParameters::WHITE_BALANCE_SHADE) ?
1895 ANDROID_CONTROL_AWB_SHADE :
1896 -1;
1897}
1898
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07001899const char* Parameters::wbModeEnumToString(uint8_t wbMode) {
1900 switch (wbMode) {
1901 case ANDROID_CONTROL_AWB_AUTO:
1902 return CameraParameters::WHITE_BALANCE_AUTO;
1903 case ANDROID_CONTROL_AWB_INCANDESCENT:
1904 return CameraParameters::WHITE_BALANCE_INCANDESCENT;
1905 case ANDROID_CONTROL_AWB_FLUORESCENT:
1906 return CameraParameters::WHITE_BALANCE_FLUORESCENT;
1907 case ANDROID_CONTROL_AWB_WARM_FLUORESCENT:
1908 return CameraParameters::WHITE_BALANCE_WARM_FLUORESCENT;
1909 case ANDROID_CONTROL_AWB_DAYLIGHT:
1910 return CameraParameters::WHITE_BALANCE_DAYLIGHT;
1911 case ANDROID_CONTROL_AWB_CLOUDY_DAYLIGHT:
1912 return CameraParameters::WHITE_BALANCE_CLOUDY_DAYLIGHT;
1913 case ANDROID_CONTROL_AWB_TWILIGHT:
1914 return CameraParameters::WHITE_BALANCE_TWILIGHT;
1915 case ANDROID_CONTROL_AWB_SHADE:
1916 return CameraParameters::WHITE_BALANCE_SHADE;
1917 default:
1918 ALOGE("%s: Unknown AWB mode enum: %d",
1919 __FUNCTION__, wbMode);
1920 return "unknown";
1921 }
1922}
1923
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001924int Parameters::effectModeStringToEnum(const char *effectMode) {
1925 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07001926 !effectMode ?
1927 ANDROID_CONTROL_EFFECT_OFF :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001928 !strcmp(effectMode, CameraParameters::EFFECT_NONE) ?
1929 ANDROID_CONTROL_EFFECT_OFF :
1930 !strcmp(effectMode, CameraParameters::EFFECT_MONO) ?
1931 ANDROID_CONTROL_EFFECT_MONO :
1932 !strcmp(effectMode, CameraParameters::EFFECT_NEGATIVE) ?
1933 ANDROID_CONTROL_EFFECT_NEGATIVE :
1934 !strcmp(effectMode, CameraParameters::EFFECT_SOLARIZE) ?
1935 ANDROID_CONTROL_EFFECT_SOLARIZE :
1936 !strcmp(effectMode, CameraParameters::EFFECT_SEPIA) ?
1937 ANDROID_CONTROL_EFFECT_SEPIA :
1938 !strcmp(effectMode, CameraParameters::EFFECT_POSTERIZE) ?
1939 ANDROID_CONTROL_EFFECT_POSTERIZE :
1940 !strcmp(effectMode, CameraParameters::EFFECT_WHITEBOARD) ?
1941 ANDROID_CONTROL_EFFECT_WHITEBOARD :
1942 !strcmp(effectMode, CameraParameters::EFFECT_BLACKBOARD) ?
1943 ANDROID_CONTROL_EFFECT_BLACKBOARD :
1944 !strcmp(effectMode, CameraParameters::EFFECT_AQUA) ?
1945 ANDROID_CONTROL_EFFECT_AQUA :
1946 -1;
1947}
1948
1949int Parameters::abModeStringToEnum(const char *abMode) {
1950 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07001951 !abMode ?
1952 ANDROID_CONTROL_AE_ANTIBANDING_AUTO :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001953 !strcmp(abMode, CameraParameters::ANTIBANDING_AUTO) ?
1954 ANDROID_CONTROL_AE_ANTIBANDING_AUTO :
1955 !strcmp(abMode, CameraParameters::ANTIBANDING_OFF) ?
1956 ANDROID_CONTROL_AE_ANTIBANDING_OFF :
1957 !strcmp(abMode, CameraParameters::ANTIBANDING_50HZ) ?
1958 ANDROID_CONTROL_AE_ANTIBANDING_50HZ :
1959 !strcmp(abMode, CameraParameters::ANTIBANDING_60HZ) ?
1960 ANDROID_CONTROL_AE_ANTIBANDING_60HZ :
1961 -1;
1962}
1963
1964int Parameters::sceneModeStringToEnum(const char *sceneMode) {
1965 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07001966 !sceneMode ?
1967 ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07001968 !strcmp(sceneMode, CameraParameters::SCENE_MODE_AUTO) ?
1969 ANDROID_CONTROL_SCENE_MODE_UNSUPPORTED :
1970 !strcmp(sceneMode, CameraParameters::SCENE_MODE_ACTION) ?
1971 ANDROID_CONTROL_SCENE_MODE_ACTION :
1972 !strcmp(sceneMode, CameraParameters::SCENE_MODE_PORTRAIT) ?
1973 ANDROID_CONTROL_SCENE_MODE_PORTRAIT :
1974 !strcmp(sceneMode, CameraParameters::SCENE_MODE_LANDSCAPE) ?
1975 ANDROID_CONTROL_SCENE_MODE_LANDSCAPE :
1976 !strcmp(sceneMode, CameraParameters::SCENE_MODE_NIGHT) ?
1977 ANDROID_CONTROL_SCENE_MODE_NIGHT :
1978 !strcmp(sceneMode, CameraParameters::SCENE_MODE_NIGHT_PORTRAIT) ?
1979 ANDROID_CONTROL_SCENE_MODE_NIGHT_PORTRAIT :
1980 !strcmp(sceneMode, CameraParameters::SCENE_MODE_THEATRE) ?
1981 ANDROID_CONTROL_SCENE_MODE_THEATRE :
1982 !strcmp(sceneMode, CameraParameters::SCENE_MODE_BEACH) ?
1983 ANDROID_CONTROL_SCENE_MODE_BEACH :
1984 !strcmp(sceneMode, CameraParameters::SCENE_MODE_SNOW) ?
1985 ANDROID_CONTROL_SCENE_MODE_SNOW :
1986 !strcmp(sceneMode, CameraParameters::SCENE_MODE_SUNSET) ?
1987 ANDROID_CONTROL_SCENE_MODE_SUNSET :
1988 !strcmp(sceneMode, CameraParameters::SCENE_MODE_STEADYPHOTO) ?
1989 ANDROID_CONTROL_SCENE_MODE_STEADYPHOTO :
1990 !strcmp(sceneMode, CameraParameters::SCENE_MODE_FIREWORKS) ?
1991 ANDROID_CONTROL_SCENE_MODE_FIREWORKS :
1992 !strcmp(sceneMode, CameraParameters::SCENE_MODE_SPORTS) ?
1993 ANDROID_CONTROL_SCENE_MODE_SPORTS :
1994 !strcmp(sceneMode, CameraParameters::SCENE_MODE_PARTY) ?
1995 ANDROID_CONTROL_SCENE_MODE_PARTY :
1996 !strcmp(sceneMode, CameraParameters::SCENE_MODE_CANDLELIGHT) ?
1997 ANDROID_CONTROL_SCENE_MODE_CANDLELIGHT :
1998 !strcmp(sceneMode, CameraParameters::SCENE_MODE_BARCODE) ?
1999 ANDROID_CONTROL_SCENE_MODE_BARCODE:
2000 -1;
2001}
2002
2003Parameters::Parameters::flashMode_t Parameters::flashModeStringToEnum(
2004 const char *flashMode) {
2005 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07002006 !flashMode ?
2007 Parameters::FLASH_MODE_INVALID :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002008 !strcmp(flashMode, CameraParameters::FLASH_MODE_OFF) ?
2009 Parameters::FLASH_MODE_OFF :
2010 !strcmp(flashMode, CameraParameters::FLASH_MODE_AUTO) ?
2011 Parameters::FLASH_MODE_AUTO :
2012 !strcmp(flashMode, CameraParameters::FLASH_MODE_ON) ?
2013 Parameters::FLASH_MODE_ON :
2014 !strcmp(flashMode, CameraParameters::FLASH_MODE_RED_EYE) ?
2015 Parameters::FLASH_MODE_RED_EYE :
2016 !strcmp(flashMode, CameraParameters::FLASH_MODE_TORCH) ?
2017 Parameters::FLASH_MODE_TORCH :
2018 Parameters::FLASH_MODE_INVALID;
2019}
2020
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07002021const char *Parameters::flashModeEnumToString(flashMode_t flashMode) {
2022 switch (flashMode) {
2023 case FLASH_MODE_OFF:
2024 return CameraParameters::FLASH_MODE_OFF;
2025 case FLASH_MODE_AUTO:
2026 return CameraParameters::FLASH_MODE_AUTO;
2027 case FLASH_MODE_ON:
2028 return CameraParameters::FLASH_MODE_ON;
2029 case FLASH_MODE_RED_EYE:
2030 return CameraParameters::FLASH_MODE_RED_EYE;
2031 case FLASH_MODE_TORCH:
2032 return CameraParameters::FLASH_MODE_TORCH;
2033 default:
2034 ALOGE("%s: Unknown flash mode enum %d",
2035 __FUNCTION__, flashMode);
2036 return "unknown";
2037 }
2038}
2039
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002040Parameters::Parameters::focusMode_t Parameters::focusModeStringToEnum(
2041 const char *focusMode) {
2042 return
Eino-Ville Talvala4a02cc82012-09-10 09:53:09 -07002043 !focusMode ?
2044 Parameters::FOCUS_MODE_INVALID :
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002045 !strcmp(focusMode, CameraParameters::FOCUS_MODE_AUTO) ?
2046 Parameters::FOCUS_MODE_AUTO :
2047 !strcmp(focusMode, CameraParameters::FOCUS_MODE_INFINITY) ?
2048 Parameters::FOCUS_MODE_INFINITY :
2049 !strcmp(focusMode, CameraParameters::FOCUS_MODE_MACRO) ?
2050 Parameters::FOCUS_MODE_MACRO :
2051 !strcmp(focusMode, CameraParameters::FOCUS_MODE_FIXED) ?
2052 Parameters::FOCUS_MODE_FIXED :
2053 !strcmp(focusMode, CameraParameters::FOCUS_MODE_EDOF) ?
2054 Parameters::FOCUS_MODE_EDOF :
2055 !strcmp(focusMode, CameraParameters::FOCUS_MODE_CONTINUOUS_VIDEO) ?
2056 Parameters::FOCUS_MODE_CONTINUOUS_VIDEO :
2057 !strcmp(focusMode, CameraParameters::FOCUS_MODE_CONTINUOUS_PICTURE) ?
2058 Parameters::FOCUS_MODE_CONTINUOUS_PICTURE :
2059 Parameters::FOCUS_MODE_INVALID;
2060}
2061
Eino-Ville Talvala8a42dd82012-10-02 13:30:04 -07002062const char *Parameters::focusModeEnumToString(focusMode_t focusMode) {
2063 switch (focusMode) {
2064 case FOCUS_MODE_AUTO:
2065 return CameraParameters::FOCUS_MODE_AUTO;
2066 case FOCUS_MODE_MACRO:
2067 return CameraParameters::FOCUS_MODE_MACRO;
2068 case FOCUS_MODE_CONTINUOUS_VIDEO:
2069 return CameraParameters::FOCUS_MODE_CONTINUOUS_VIDEO;
2070 case FOCUS_MODE_CONTINUOUS_PICTURE:
2071 return CameraParameters::FOCUS_MODE_CONTINUOUS_PICTURE;
2072 case FOCUS_MODE_EDOF:
2073 return CameraParameters::FOCUS_MODE_EDOF;
2074 case FOCUS_MODE_INFINITY:
2075 return CameraParameters::FOCUS_MODE_INFINITY;
2076 case FOCUS_MODE_FIXED:
2077 return CameraParameters::FOCUS_MODE_FIXED;
2078 default:
2079 ALOGE("%s: Unknown focus mode enum: %d",
2080 __FUNCTION__, focusMode);
2081 return "unknown";
2082 }
2083}
2084
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002085status_t Parameters::parseAreas(const char *areasCStr,
2086 Vector<Parameters::Area> *areas) {
2087 static const size_t NUM_FIELDS = 5;
2088 areas->clear();
2089 if (areasCStr == NULL) {
2090 // If no key exists, use default (0,0,0,0,0)
2091 areas->push();
2092 return OK;
2093 }
2094 String8 areasStr(areasCStr);
2095 ssize_t areaStart = areasStr.find("(", 0) + 1;
2096 while (areaStart != 0) {
2097 const char* area = areasStr.string() + areaStart;
2098 char *numEnd;
2099 int vals[NUM_FIELDS];
2100 for (size_t i = 0; i < NUM_FIELDS; i++) {
2101 errno = 0;
2102 vals[i] = strtol(area, &numEnd, 10);
2103 if (errno || numEnd == area) return BAD_VALUE;
2104 area = numEnd + 1;
2105 }
2106 areas->push(Parameters::Area(
2107 vals[0], vals[1], vals[2], vals[3], vals[4]) );
2108 areaStart = areasStr.find("(", areaStart) + 1;
2109 }
2110 return OK;
2111}
2112
2113status_t Parameters::validateAreas(const Vector<Parameters::Area> &areas,
2114 size_t maxRegions) {
2115 // Definition of valid area can be found in
2116 // include/camera/CameraParameters.h
2117 if (areas.size() == 0) return BAD_VALUE;
2118 if (areas.size() == 1) {
2119 if (areas[0].left == 0 &&
2120 areas[0].top == 0 &&
2121 areas[0].right == 0 &&
2122 areas[0].bottom == 0 &&
2123 areas[0].weight == 0) {
2124 // Single (0,0,0,0,0) entry is always valid (== driver decides)
2125 return OK;
2126 }
2127 }
2128 if (areas.size() > maxRegions) {
2129 ALOGE("%s: Too many areas requested: %d",
2130 __FUNCTION__, areas.size());
2131 return BAD_VALUE;
2132 }
2133
2134 for (Vector<Parameters::Area>::const_iterator a = areas.begin();
2135 a != areas.end(); a++) {
2136 if (a->weight < 1 || a->weight > 1000) return BAD_VALUE;
2137 if (a->left < -1000 || a->left > 1000) return BAD_VALUE;
2138 if (a->top < -1000 || a->top > 1000) return BAD_VALUE;
2139 if (a->right < -1000 || a->right > 1000) return BAD_VALUE;
2140 if (a->bottom < -1000 || a->bottom > 1000) return BAD_VALUE;
2141 if (a->left >= a->right) return BAD_VALUE;
2142 if (a->top >= a->bottom) return BAD_VALUE;
2143 }
2144 return OK;
2145}
2146
2147bool Parameters::boolFromString(const char *boolStr) {
2148 return !boolStr ? false :
2149 !strcmp(boolStr, CameraParameters::TRUE) ? true :
2150 false;
2151}
2152
2153int Parameters::degToTransform(int degrees, bool mirror) {
2154 if (!mirror) {
2155 if (degrees == 0) return 0;
2156 else if (degrees == 90) return HAL_TRANSFORM_ROT_90;
2157 else if (degrees == 180) return HAL_TRANSFORM_ROT_180;
2158 else if (degrees == 270) return HAL_TRANSFORM_ROT_270;
2159 } else { // Do mirror (horizontal flip)
2160 if (degrees == 0) { // FLIP_H and ROT_0
2161 return HAL_TRANSFORM_FLIP_H;
2162 } else if (degrees == 90) { // FLIP_H and ROT_90
2163 return HAL_TRANSFORM_FLIP_H | HAL_TRANSFORM_ROT_90;
2164 } else if (degrees == 180) { // FLIP_H and ROT_180
2165 return HAL_TRANSFORM_FLIP_V;
2166 } else if (degrees == 270) { // FLIP_H and ROT_270
2167 return HAL_TRANSFORM_FLIP_V | HAL_TRANSFORM_ROT_90;
2168 }
2169 }
2170 ALOGE("%s: Bad input: %d", __FUNCTION__, degrees);
2171 return -1;
2172}
2173
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002174int Parameters::cropXToArray(int x) const {
2175 ALOG_ASSERT(x >= 0, "Crop-relative X coordinate = '%d' is out of bounds"
2176 "(lower = 0)", x);
2177
2178 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2179 ALOG_ASSERT(x < previewCrop.width, "Crop-relative X coordinate = '%d' "
2180 "is out of bounds (upper = %d)", x, previewCrop.width);
2181
2182 int ret = x + previewCrop.left;
2183
2184 ALOG_ASSERT( (ret >= 0 && ret < fastInfo.arrayWidth),
2185 "Calculated pixel array value X = '%d' is out of bounds (upper = %d)",
2186 ret, fastInfo.arrayWidth);
2187 return ret;
2188}
2189
2190int Parameters::cropYToArray(int y) const {
2191 ALOG_ASSERT(y >= 0, "Crop-relative Y coordinate = '%d' is out of bounds "
2192 "(lower = 0)", y);
2193
2194 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2195 ALOG_ASSERT(y < previewCrop.height, "Crop-relative Y coordinate = '%d' is "
2196 "out of bounds (upper = %d)", y, previewCrop.height);
2197
2198 int ret = y + previewCrop.top;
2199
2200 ALOG_ASSERT( (ret >= 0 && ret < fastInfo.arrayHeight),
2201 "Calculated pixel array value Y = '%d' is out of bounds (upper = %d)",
2202 ret, fastInfo.arrayHeight);
2203
2204 return ret;
2205
2206}
2207
2208int Parameters::normalizedXToCrop(int x) const {
2209 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2210 return (x + 1000) * (previewCrop.width - 1) / 2000;
2211}
2212
2213int Parameters::normalizedYToCrop(int y) const {
2214 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2215 return (y + 1000) * (previewCrop.height - 1) / 2000;
2216}
2217
2218int Parameters::arrayXToCrop(int x) const {
2219 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2220 return x - previewCrop.left;
2221}
2222
2223int Parameters::arrayYToCrop(int y) const {
2224 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2225 return y - previewCrop.top;
2226}
2227
2228int Parameters::cropXToNormalized(int x) const {
2229 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2230 return x * 2000 / (previewCrop.width - 1) - 1000;
2231}
2232
2233int Parameters::cropYToNormalized(int y) const {
2234 CropRegion previewCrop = calculateCropRegion(CropRegion::OUTPUT_PREVIEW);
2235 return y * 2000 / (previewCrop.height - 1) - 1000;
2236}
2237
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002238int Parameters::arrayXToNormalized(int width) const {
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002239 int ret = cropXToNormalized(arrayXToCrop(width));
2240
2241 ALOG_ASSERT(ret >= -1000, "Calculated normalized value out of "
2242 "lower bounds %d", ret);
2243 ALOG_ASSERT(ret <= 1000, "Calculated normalized value out of "
2244 "upper bounds %d", ret);
2245
2246 // Work-around for HAL pre-scaling the coordinates themselves
2247 if (quirks.meteringCropRegion) {
2248 return width * 2000 / (fastInfo.arrayWidth - 1) - 1000;
2249 }
2250
2251 return ret;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002252}
2253
2254int Parameters::arrayYToNormalized(int height) const {
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002255 int ret = cropYToNormalized(arrayYToCrop(height));
2256
2257 ALOG_ASSERT(ret >= -1000, "Calculated normalized value out of lower bounds"
2258 " %d", ret);
2259 ALOG_ASSERT(ret <= 1000, "Calculated normalized value out of upper bounds"
2260 " %d", ret);
2261
2262 // Work-around for HAL pre-scaling the coordinates themselves
2263 if (quirks.meteringCropRegion) {
2264 return height * 2000 / (fastInfo.arrayHeight - 1) - 1000;
2265 }
2266
2267 return ret;
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002268}
2269
2270int Parameters::normalizedXToArray(int x) const {
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002271
2272 // Work-around for HAL pre-scaling the coordinates themselves
2273 if (quirks.meteringCropRegion) {
2274 return (x + 1000) * (fastInfo.arrayWidth - 1) / 2000;
2275 }
2276
2277 return cropXToArray(normalizedXToCrop(x));
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002278}
2279
2280int Parameters::normalizedYToArray(int y) const {
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002281 // Work-around for HAL pre-scaling the coordinates themselves
2282 if (quirks.meteringCropRegion) {
2283 return (y + 1000) * (fastInfo.arrayHeight - 1) / 2000;
2284 }
2285
2286 return cropYToArray(normalizedYToCrop(y));
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002287}
2288
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002289Parameters::CropRegion Parameters::calculateCropRegion(
2290 Parameters::CropRegion::Outputs outputs) const {
Igor Murashkin018d2282012-09-18 18:23:49 -07002291
2292 float zoomLeft, zoomTop, zoomWidth, zoomHeight;
2293
2294 // Need to convert zoom index into a crop rectangle. The rectangle is
2295 // chosen to maximize its area on the sensor
2296
2297 camera_metadata_ro_entry_t maxDigitalZoom =
2298 staticInfo(ANDROID_SCALER_AVAILABLE_MAX_ZOOM);
2299 // For each zoom step by how many pixels more do we change the zoom
2300 float zoomIncrement = (maxDigitalZoom.data.f[0] - 1) /
2301 (NUM_ZOOM_STEPS-1);
2302 // The desired activeAreaWidth/cropAreaWidth ratio (or height if h>w)
2303 // via interpolating zoom step into a zoom ratio
2304 float zoomRatio = 1 + zoomIncrement * zoom;
2305 ALOG_ASSERT( (zoomRatio >= 1.f && zoomRatio <= maxDigitalZoom.data.f[0]),
2306 "Zoom ratio calculated out of bounds. Expected 1 - %f, actual: %f",
2307 maxDigitalZoom.data.f[0], zoomRatio);
2308
2309 ALOGV("Zoom maxDigital=%f, increment=%f, ratio=%f, previewWidth=%d, "
2310 "previewHeight=%d, activeWidth=%d, activeHeight=%d",
2311 maxDigitalZoom.data.f[0], zoomIncrement, zoomRatio, previewWidth,
2312 previewHeight, fastInfo.arrayWidth, fastInfo.arrayHeight);
2313
2314 /*
2315 * Assumption: On the HAL side each stream buffer calculates its crop
2316 * rectangle as follows:
2317 * cropRect = (zoomLeft, zoomRight,
2318 * zoomWidth, zoomHeight * zoomWidth / outputWidth);
2319 *
2320 * Note that if zoomWidth > bufferWidth, the new cropHeight > zoomHeight
2321 * (we can then get into trouble if the cropHeight > arrayHeight).
2322 * By selecting the zoomRatio based on the smallest outputRatio, we
2323 * guarantee this will never happen.
2324 */
2325
2326 // Enumerate all possible output sizes, select the one with the smallest
2327 // aspect ratio
2328 float minOutputWidth, minOutputHeight, minOutputRatio;
2329 {
2330 float outputSizes[][2] = {
2331 { previewWidth, previewHeight },
2332 { videoWidth, videoHeight },
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002333 { jpegThumbSize[0], jpegThumbSize[1] },
Igor Murashkin018d2282012-09-18 18:23:49 -07002334 { pictureWidth, pictureHeight },
2335 };
2336
2337 minOutputWidth = outputSizes[0][0];
2338 minOutputHeight = outputSizes[0][1];
2339 minOutputRatio = minOutputWidth / minOutputHeight;
2340 for (unsigned int i = 0;
2341 i < sizeof(outputSizes) / sizeof(outputSizes[0]);
2342 ++i) {
2343
Igor Murashkin7373cbe2012-09-28 15:30:03 -07002344 // skip over outputs we don't want to consider for the crop region
2345 if ( !((1 << i) & outputs) ) {
2346 continue;
2347 }
2348
Igor Murashkin018d2282012-09-18 18:23:49 -07002349 float outputWidth = outputSizes[i][0];
2350 float outputHeight = outputSizes[i][1];
2351 float outputRatio = outputWidth / outputHeight;
2352
2353 if (minOutputRatio > outputRatio) {
2354 minOutputRatio = outputRatio;
2355 minOutputWidth = outputWidth;
2356 minOutputHeight = outputHeight;
2357 }
2358
2359 // and then use this output ratio instead of preview output ratio
2360 ALOGV("Enumerating output ratio %f = %f / %f, min is %f",
2361 outputRatio, outputWidth, outputHeight, minOutputRatio);
2362 }
2363 }
2364
2365 /* Ensure that the width/height never go out of bounds
2366 * by scaling across a diffent dimension if an out-of-bounds
2367 * possibility exists.
2368 *
2369 * e.g. if the previewratio < arrayratio and e.g. zoomratio = 1.0, then by
2370 * calculating the zoomWidth from zoomHeight we'll actually get a
2371 * zoomheight > arrayheight
2372 */
2373 float arrayRatio = 1.f * fastInfo.arrayWidth / fastInfo.arrayHeight;
2374 if (minOutputRatio >= arrayRatio) {
2375 // Adjust the height based on the width
2376 zoomWidth = fastInfo.arrayWidth / zoomRatio;
2377 zoomHeight = zoomWidth *
2378 minOutputHeight / minOutputWidth;
2379
2380 } else {
2381 // Adjust the width based on the height
2382 zoomHeight = fastInfo.arrayHeight / zoomRatio;
2383 zoomWidth = zoomHeight *
2384 minOutputWidth / minOutputHeight;
2385 }
2386 // centering the zoom area within the active area
2387 zoomLeft = (fastInfo.arrayWidth - zoomWidth) / 2;
2388 zoomTop = (fastInfo.arrayHeight - zoomHeight) / 2;
2389
2390 ALOGV("Crop region calculated (x=%d,y=%d,w=%f,h=%f) for zoom=%d",
2391 (int32_t)zoomLeft, (int32_t)zoomTop, zoomWidth, zoomHeight, this->zoom);
2392
2393
2394 CropRegion crop = { zoomLeft, zoomTop, zoomWidth, zoomHeight };
2395 return crop;
2396}
2397
Igor Murashkinaf3d2882012-10-04 14:22:18 -07002398int32_t Parameters::fpsFromRange(int32_t min, int32_t max) const {
2399 return max;
2400}
2401
Eino-Ville Talvala2e19c3c2012-08-26 09:29:28 -07002402}; // namespace camera2
2403}; // namespace android