1 /*
2 * Copyright (C) 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 //#define LOG_NDEBUG 0
18 #define LOG_TAG "GCH_RgbirdRtRequestProcessor"
19 #define ATRACE_TAG ATRACE_TAG_CAMERA
20 #include <cutils/properties.h>
21 #include <log/log.h>
22 #include <utils/Trace.h>
23
24 #include "hal_utils.h"
25 #include "rgbird_rt_request_processor.h"
26 #include "vendor_tag_defs.h"
27
28 namespace android {
29 namespace google_camera_hal {
30
Create(CameraDeviceSessionHwl * device_session_hwl,bool is_hdrplus_supported)31 std::unique_ptr<RgbirdRtRequestProcessor> RgbirdRtRequestProcessor::Create(
32 CameraDeviceSessionHwl* device_session_hwl, bool is_hdrplus_supported) {
33 ATRACE_CALL();
34 if (device_session_hwl == nullptr) {
35 ALOGE("%s: device_session_hwl is nullptr", __FUNCTION__);
36 return nullptr;
37 }
38
39 std::vector<uint32_t> physical_camera_ids =
40 device_session_hwl->GetPhysicalCameraIds();
41 if (physical_camera_ids.size() != 3) {
42 ALOGE("%s: Only support 3 cameras", __FUNCTION__);
43 return nullptr;
44 }
45
46 std::unique_ptr<HalCameraMetadata> characteristics;
47 status_t res = device_session_hwl->GetCameraCharacteristics(&characteristics);
48 if (res != OK) {
49 ALOGE("%s: GetCameraCharacteristics failed.", __FUNCTION__);
50 return nullptr;
51 }
52
53 uint32_t active_array_width, active_array_height;
54 camera_metadata_ro_entry entry;
55 res = characteristics->Get(
56 ANDROID_SENSOR_INFO_PRE_CORRECTION_ACTIVE_ARRAY_SIZE, &entry);
57 if (res == OK) {
58 active_array_width = entry.data.i32[2];
59 active_array_height = entry.data.i32[3];
60 ALOGI("%s Active size (%d x %d).", __FUNCTION__, active_array_width,
61 active_array_height);
62 } else {
63 ALOGE("%s Get active size failed: %s (%d).", __FUNCTION__, strerror(-res),
64 res);
65 return nullptr;
66 }
67
68 auto request_processor =
69 std::unique_ptr<RgbirdRtRequestProcessor>(new RgbirdRtRequestProcessor(
70 physical_camera_ids[0], physical_camera_ids[1],
71 physical_camera_ids[2], active_array_width, active_array_height,
72 is_hdrplus_supported, device_session_hwl));
73 if (request_processor == nullptr) {
74 ALOGE("%s: Creating RgbirdRtRequestProcessor failed.", __FUNCTION__);
75 return nullptr;
76 }
77
78 // TODO(b/128633958): remove this after FLL syncing is verified
79 request_processor->force_internal_stream_ =
80 property_get_bool("persist.vendor.camera.rgbird.forceinternal", false);
81 if (request_processor->force_internal_stream_) {
82 ALOGI("%s: Force creating internal streams for IR pipelines", __FUNCTION__);
83 }
84
85 // TODO(b/129910835): This prop should be removed once that logic is in place.
86 request_processor->rgb_ir_auto_cal_enabled_ =
87 property_get_bool("vendor.camera.frontdepth.enableautocal", true);
88 if (request_processor->rgb_ir_auto_cal_enabled_) {
89 ALOGI("%s: ", __FUNCTION__);
90 }
91 request_processor->is_auto_cal_session_ =
92 request_processor->IsAutocalSession();
93
94 return request_processor;
95 }
96
IsAutocalSession() const97 bool RgbirdRtRequestProcessor::IsAutocalSession() const {
98 // TODO(b/129910835): Use more specific logic to determine if a session needs
99 // to run autocal or not. Even if rgb_ir_auto_cal_enabled_ is true, it is
100 // more reasonable to only run auto cal for some sessions(e.g. 1st session
101 // after device boot that has a depth stream configured).
102 // To allow more tests, every session having a depth stream is an autocal
103 // session now.
104 return rgb_ir_auto_cal_enabled_;
105 }
106
IsAutocalRequest(uint32_t frame_number)107 bool RgbirdRtRequestProcessor::IsAutocalRequest(uint32_t frame_number) {
108 // TODO(b/129910835): Refine the logic here to only trigger auto cal for
109 // specific request. The result/request processor and depth process block has
110 // final right to determine if an internal yuv stream buffer will be used for
111 // autocal.
112 // The current logic is to trigger the autocal in the kAutocalFrameNumber
113 // frame. This must be consistent with that of result_request_processor.
114 if (!is_auto_cal_session_ || auto_cal_triggered_ ||
115 frame_number != kAutocalFrameNumber ||
116 depth_stream_id_ == kStreamIdInvalid) {
117 return false;
118 }
119
120 auto_cal_triggered_ = true;
121 return true;
122 }
123
RgbirdRtRequestProcessor(uint32_t rgb_camera_id,uint32_t ir1_camera_id,uint32_t ir2_camera_id,uint32_t active_array_width,uint32_t active_array_height,bool is_hdrplus_supported,CameraDeviceSessionHwl * device_session_hwl)124 RgbirdRtRequestProcessor::RgbirdRtRequestProcessor(
125 uint32_t rgb_camera_id, uint32_t ir1_camera_id, uint32_t ir2_camera_id,
126 uint32_t active_array_width, uint32_t active_array_height,
127 bool is_hdrplus_supported, CameraDeviceSessionHwl* device_session_hwl)
128 : kRgbCameraId(rgb_camera_id),
129 kIr1CameraId(ir1_camera_id),
130 kIr2CameraId(ir2_camera_id),
131 rgb_active_array_width_(active_array_width),
132 rgb_active_array_height_(active_array_height),
133 is_hdrplus_supported_(is_hdrplus_supported),
134 is_hdrplus_zsl_enabled_(is_hdrplus_supported),
135 device_session_hwl_(device_session_hwl) {
136 ALOGI(
137 "%s: Created a RGBIRD RT request processor for RGB %u, IR1 %u, IR2 %u, "
138 "is_hdrplus_supported_ :%d",
139 __FUNCTION__, kRgbCameraId, kIr1CameraId, kIr2CameraId,
140 is_hdrplus_supported_);
141 }
142
FindSmallestNonWarpedYuvStreamResolution(uint32_t * yuv_w_adjusted,uint32_t * yuv_h_adjusted)143 status_t RgbirdRtRequestProcessor::FindSmallestNonWarpedYuvStreamResolution(
144 uint32_t* yuv_w_adjusted, uint32_t* yuv_h_adjusted) {
145 if (yuv_w_adjusted == nullptr || yuv_h_adjusted == nullptr) {
146 ALOGE("%s: yuv_w_adjusted or yuv_h_adjusted is nullptr.", __FUNCTION__);
147 return BAD_VALUE;
148 }
149
150 std::unique_ptr<HalCameraMetadata> characteristics;
151 status_t res = device_session_hwl_->GetCameraCharacteristics(&characteristics);
152 if (res != OK) {
153 ALOGE("%s: GetCameraCharacteristics failed.", __FUNCTION__);
154 return UNKNOWN_ERROR;
155 }
156 camera_metadata_ro_entry entry;
157 res = characteristics->Get(VendorTagIds::kAvailableNonWarpedYuvSizes, &entry);
158 if (res != OK) {
159 ALOGE("%s Get stream size failed: %s (%d).", __FUNCTION__, strerror(-res),
160 res);
161 return UNKNOWN_ERROR;
162 }
163
164 uint32_t min_area = std::numeric_limits<uint32_t>::max();
165 uint32_t current_area = 0;
166 for (size_t i = 0; i < entry.count; i += 2) {
167 current_area = entry.data.i32[i] * entry.data.i32[i + 1];
168 if (current_area < min_area) {
169 *yuv_w_adjusted = entry.data.i32[i];
170 *yuv_h_adjusted = entry.data.i32[i + 1];
171 min_area = current_area;
172 }
173 }
174
175 return OK;
176 }
177
FindSmallestResolutionForInternalYuvStream(const StreamConfiguration & process_block_stream_config,uint32_t * yuv_w_adjusted,uint32_t * yuv_h_adjusted)178 status_t RgbirdRtRequestProcessor::FindSmallestResolutionForInternalYuvStream(
179 const StreamConfiguration& process_block_stream_config,
180 uint32_t* yuv_w_adjusted, uint32_t* yuv_h_adjusted) {
181 if (yuv_w_adjusted == nullptr || yuv_h_adjusted == nullptr) {
182 ALOGE("%s: yuv_w_adjusted or yuv_h_adjusted is nullptr.", __FUNCTION__);
183 return BAD_VALUE;
184 }
185
186 *yuv_w_adjusted = kDefaultYuvStreamWidth;
187 *yuv_h_adjusted = kDefaultYuvStreamHeight;
188 uint32_t framework_non_raw_w = 0;
189 uint32_t framework_non_raw_h = 0;
190 bool non_raw_non_depth_stream_configured = false;
191 for (auto& stream : process_block_stream_config.streams) {
192 if (!utils::IsRawStream(stream) && !utils::IsDepthStream(stream)) {
193 non_raw_non_depth_stream_configured = true;
194 framework_non_raw_w = stream.width;
195 framework_non_raw_h = stream.height;
196 break;
197 }
198 }
199
200 std::unique_ptr<HalCameraMetadata> characteristics;
201 status_t res = device_session_hwl_->GetCameraCharacteristics(&characteristics);
202 if (res != OK) {
203 ALOGE("%s: GetCameraCharacteristics failed.", __FUNCTION__);
204 return UNKNOWN_ERROR;
205 }
206 camera_metadata_ro_entry entry;
207 res = characteristics->Get(ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS,
208 &entry);
209 if (res != OK) {
210 ALOGE("%s Get stream size failed: %s (%d).", __FUNCTION__, strerror(-res),
211 res);
212 return UNKNOWN_ERROR;
213 }
214
215 uint32_t min_area = std::numeric_limits<uint32_t>::max();
216 uint32_t current_area = 0;
217 if (non_raw_non_depth_stream_configured) {
218 bool found_matching_aspect_ratio = false;
219 for (size_t i = 0; i < entry.count; i += 4) {
220 uint8_t format = entry.data.i32[i];
221 if ((format == HAL_PIXEL_FORMAT_YCbCr_420_888) &&
222 (entry.data.i32[i + 3] ==
223 ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS_OUTPUT)) {
224 current_area = entry.data.i32[i + 1] * entry.data.i32[i + 2];
225 if ((entry.data.i32[i + 1] * framework_non_raw_h ==
226 entry.data.i32[i + 2] * framework_non_raw_w) &&
227 (current_area < min_area)) {
228 *yuv_w_adjusted = entry.data.i32[i + 1];
229 *yuv_h_adjusted = entry.data.i32[i + 2];
230 min_area = current_area;
231 found_matching_aspect_ratio = true;
232 }
233 }
234 }
235 if (!found_matching_aspect_ratio) {
236 ALOGE(
237 "%s: No matching aspect ratio can be found in the available stream"
238 "config resolution list.",
239 __FUNCTION__);
240 return UNKNOWN_ERROR;
241 }
242 } else {
243 ALOGI(
244 "No YUV stream configured, ues smallest resolution for internal "
245 "stream.");
246 for (size_t i = 0; i < entry.count; i += 4) {
247 if ((entry.data.i32[i] == HAL_PIXEL_FORMAT_YCbCr_420_888) &&
248 (entry.data.i32[i + 3] ==
249 ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS_OUTPUT)) {
250 current_area = entry.data.i32[i + 1] * entry.data.i32[i + 2];
251 if (current_area < min_area) {
252 *yuv_w_adjusted = entry.data.i32[i + 1];
253 *yuv_h_adjusted = entry.data.i32[i + 2];
254 min_area = current_area;
255 }
256 }
257 }
258 }
259
260 if ((*yuv_w_adjusted == 0) || (*yuv_h_adjusted == 0)) {
261 ALOGE("%s Get internal YUV stream size failed.", __FUNCTION__);
262 return UNKNOWN_ERROR;
263 }
264
265 return OK;
266 }
267
SetNonWarpedYuvStreamId(int32_t non_warped_yuv_stream_id,StreamConfiguration * process_block_stream_config)268 status_t RgbirdRtRequestProcessor::SetNonWarpedYuvStreamId(
269 int32_t non_warped_yuv_stream_id,
270 StreamConfiguration* process_block_stream_config) {
271 if (process_block_stream_config == nullptr) {
272 ALOGE("%s: process_block_stream_config is nullptr.", __FUNCTION__);
273 return BAD_VALUE;
274 }
275
276 if (process_block_stream_config->session_params == nullptr) {
277 uint32_t num_entries = 128;
278 uint32_t data_bytes = 512;
279
280 process_block_stream_config->session_params =
281 HalCameraMetadata::Create(num_entries, data_bytes);
282 if (process_block_stream_config->session_params == nullptr) {
283 ALOGE("%s: Failed to create session parameter.", __FUNCTION__);
284 return UNKNOWN_ERROR;
285 }
286 }
287
288 auto logical_metadata = process_block_stream_config->session_params.get();
289
290 status_t res = logical_metadata->Set(VendorTagIds::kNonWarpedYuvStreamId,
291 &non_warped_yuv_stream_id, 1);
292 if (res != OK) {
293 ALOGE("%s: Failed to update VendorTagIds::kNonWarpedYuvStreamId: %s(%d)",
294 __FUNCTION__, strerror(-res), res);
295 return UNKNOWN_ERROR;
296 }
297
298 return res;
299 }
300
CreateDepthInternalStreams(InternalStreamManager * internal_stream_manager,StreamConfiguration * process_block_stream_config)301 status_t RgbirdRtRequestProcessor::CreateDepthInternalStreams(
302 InternalStreamManager* internal_stream_manager,
303 StreamConfiguration* process_block_stream_config) {
304 ATRACE_CALL();
305
306 uint32_t yuv_w_adjusted = 0;
307 uint32_t yuv_h_adjusted = 0;
308 status_t result = OK;
309
310 if (IsAutocalSession()) {
311 result = FindSmallestNonWarpedYuvStreamResolution(&yuv_w_adjusted,
312 &yuv_h_adjusted);
313 if (result != OK) {
314 ALOGE("%s: Could not find non-warped YUV resolution for internal YUV.",
315 __FUNCTION__);
316 return UNKNOWN_ERROR;
317 }
318 } else {
319 result = FindSmallestResolutionForInternalYuvStream(
320 *process_block_stream_config, &yuv_w_adjusted, &yuv_h_adjusted);
321 if (result != OK) {
322 ALOGE("%s: Could not find compatible resolution for internal YUV.",
323 __FUNCTION__);
324 return UNKNOWN_ERROR;
325 }
326 }
327
328 ALOGI("Depth internal YUV stream (%d x %d)", yuv_w_adjusted, yuv_h_adjusted);
329 // create internal streams:
330 // 1 YUV(must have for autocal and 3-sensor syncing)
331 // 2 RAW(must have to generate depth)
332 Stream yuv_stream;
333 yuv_stream.stream_type = StreamType::kOutput;
334 yuv_stream.width = yuv_w_adjusted;
335 yuv_stream.height = yuv_h_adjusted;
336 yuv_stream.format = HAL_PIXEL_FORMAT_YCBCR_420_888;
337 yuv_stream.usage = 0;
338 yuv_stream.rotation = StreamRotation::kRotation0;
339 yuv_stream.data_space = HAL_DATASPACE_ARBITRARY;
340 yuv_stream.is_physical_camera_stream = true;
341 yuv_stream.physical_camera_id = kRgbCameraId;
342
343 result = internal_stream_manager->RegisterNewInternalStream(
344 yuv_stream, &rgb_yuv_stream_id_);
345 if (result != OK) {
346 ALOGE("%s: RegisterNewInternalStream failed.", __FUNCTION__);
347 return UNKNOWN_ERROR;
348 }
349 yuv_stream.id = rgb_yuv_stream_id_;
350
351 if (IsAutocalSession()) {
352 result = SetNonWarpedYuvStreamId(rgb_yuv_stream_id_,
353 process_block_stream_config);
354 }
355
356 if (result != OK) {
357 ALOGE("%s: Failed to set no post processing yuv stream id.", __FUNCTION__);
358 return UNKNOWN_ERROR;
359 }
360
361 Stream raw_stream[2];
362 for (uint32_t i = 0; i < 2; i++) {
363 raw_stream[i].stream_type = StreamType::kOutput;
364 raw_stream[i].width = 640;
365 raw_stream[i].height = 480;
366 raw_stream[i].format = HAL_PIXEL_FORMAT_Y8;
367 raw_stream[i].usage = 0;
368 raw_stream[i].rotation = StreamRotation::kRotation0;
369 raw_stream[i].data_space = HAL_DATASPACE_ARBITRARY;
370 raw_stream[i].is_physical_camera_stream = true;
371
372 status_t result = internal_stream_manager->RegisterNewInternalStream(
373 raw_stream[i], &ir_raw_stream_id_[i]);
374 if (result != OK) {
375 ALOGE("%s: RegisterNewInternalStream failed.", __FUNCTION__);
376 return UNKNOWN_ERROR;
377 }
378 raw_stream[i].id = ir_raw_stream_id_[i];
379 }
380
381 raw_stream[0].physical_camera_id = kIr1CameraId;
382 raw_stream[1].physical_camera_id = kIr2CameraId;
383
384 process_block_stream_config->streams.push_back(yuv_stream);
385 process_block_stream_config->streams.push_back(raw_stream[0]);
386 process_block_stream_config->streams.push_back(raw_stream[1]);
387
388 return OK;
389 }
390
RegisterHdrplusInternalRaw(StreamConfiguration * process_block_stream_config)391 status_t RgbirdRtRequestProcessor::RegisterHdrplusInternalRaw(
392 StreamConfiguration* process_block_stream_config) {
393 ATRACE_CALL();
394 if (process_block_stream_config == nullptr) {
395 ALOGE("%s: process_block_stream_config is nullptr", __FUNCTION__);
396 return BAD_VALUE;
397 }
398
399 // Register internal raw stream
400 Stream raw_stream;
401 raw_stream.stream_type = StreamType::kOutput;
402 raw_stream.width = rgb_active_array_width_;
403 raw_stream.height = rgb_active_array_height_;
404 raw_stream.format = HAL_PIXEL_FORMAT_RAW10;
405 raw_stream.usage = 0;
406 raw_stream.rotation = StreamRotation::kRotation0;
407 raw_stream.data_space = HAL_DATASPACE_ARBITRARY;
408
409 status_t result = internal_stream_manager_->RegisterNewInternalStream(
410 raw_stream, &rgb_raw_stream_id_);
411 if (result != OK) {
412 ALOGE("%s: RegisterNewInternalStream failed.", __FUNCTION__);
413 return UNKNOWN_ERROR;
414 }
415 // Set id back to raw_stream and then HWL can get correct HAL stream ID
416 raw_stream.id = rgb_raw_stream_id_;
417
418 raw_stream.is_physical_camera_stream = true;
419 raw_stream.physical_camera_id = kRgbCameraId;
420
421 // Add internal RAW stream
422 process_block_stream_config->streams.push_back(raw_stream);
423 return OK;
424 }
425
ConfigureStreams(InternalStreamManager * internal_stream_manager,const StreamConfiguration & stream_config,StreamConfiguration * process_block_stream_config)426 status_t RgbirdRtRequestProcessor::ConfigureStreams(
427 InternalStreamManager* internal_stream_manager,
428 const StreamConfiguration& stream_config,
429 StreamConfiguration* process_block_stream_config) {
430 ATRACE_CALL();
431 if (process_block_stream_config == nullptr) {
432 ALOGE("%s: process_block_stream_config is nullptr", __FUNCTION__);
433 return BAD_VALUE;
434 }
435
436 internal_stream_manager_ = internal_stream_manager;
437 if (is_hdrplus_supported_) {
438 status_t result = RegisterHdrplusInternalRaw(process_block_stream_config);
439 if (result != OK) {
440 ALOGE("%s: RegisterHdrplusInternalRaw failed.", __FUNCTION__);
441 return UNKNOWN_ERROR;
442 }
443 }
444
445 process_block_stream_config->operation_mode = stream_config.operation_mode;
446 process_block_stream_config->session_params =
447 HalCameraMetadata::Clone(stream_config.session_params.get());
448 process_block_stream_config->stream_config_counter =
449 stream_config.stream_config_counter;
450 process_block_stream_config->log_id = stream_config.log_id;
451
452 bool has_depth_stream = false;
453 for (auto& stream : stream_config.streams) {
454 if (utils::IsDepthStream(stream)) {
455 has_depth_stream = true;
456 depth_stream_id_ = stream.id;
457 continue;
458 }
459
460 auto pb_stream = stream;
461 // Assign all logical streams to RGB camera.
462 if (!pb_stream.is_physical_camera_stream) {
463 pb_stream.is_physical_camera_stream = true;
464 pb_stream.physical_camera_id = kRgbCameraId;
465 }
466
467 process_block_stream_config->streams.push_back(pb_stream);
468 }
469
470 // TODO(b/128633958): remove the force flag after FLL syncing is verified
471 if (force_internal_stream_ || has_depth_stream) {
472 CreateDepthInternalStreams(internal_stream_manager,
473 process_block_stream_config);
474 }
475
476 return OK;
477 }
478
SetProcessBlock(std::unique_ptr<ProcessBlock> process_block)479 status_t RgbirdRtRequestProcessor::SetProcessBlock(
480 std::unique_ptr<ProcessBlock> process_block) {
481 ATRACE_CALL();
482 if (process_block == nullptr) {
483 ALOGE("%s: process_block is nullptr", __FUNCTION__);
484 return BAD_VALUE;
485 }
486
487 std::lock_guard<std::mutex> lock(process_block_lock_);
488 if (process_block_ != nullptr) {
489 ALOGE("%s: Already configured.", __FUNCTION__);
490 return ALREADY_EXISTS;
491 }
492
493 process_block_ = std::move(process_block);
494 return OK;
495 }
496
AddIrRawProcessBlockRequestLocked(std::vector<ProcessBlockRequest> * block_requests,const CaptureRequest & request,uint32_t camera_id)497 status_t RgbirdRtRequestProcessor::AddIrRawProcessBlockRequestLocked(
498 std::vector<ProcessBlockRequest>* block_requests,
499 const CaptureRequest& request, uint32_t camera_id) {
500 ATRACE_CALL();
501 uint32_t stream_id_index = 0;
502
503 if (camera_id == kIr1CameraId) {
504 stream_id_index = 0;
505 } else if (camera_id == kIr2CameraId) {
506 stream_id_index = 1;
507 } else {
508 ALOGE("%s: Unknown IR camera id %d", __FUNCTION__, camera_id);
509 return INVALID_OPERATION;
510 }
511
512 ProcessBlockRequest block_request = {.request_id = camera_id};
513 CaptureRequest& physical_request = block_request.request;
514 physical_request.frame_number = request.frame_number;
515 physical_request.settings = HalCameraMetadata::Clone(request.settings.get());
516
517 // TODO(b/128633958): Remap the crop region for IR sensors properly.
518 // The crop region cloned from logical camera control settings causes mass log
519 // spew from the IR pipelines. Force the crop region for now as a WAR.
520 if (physical_request.settings != nullptr) {
521 camera_metadata_ro_entry_t entry_crop_region_user = {};
522 if (physical_request.settings->Get(ANDROID_SCALER_CROP_REGION,
523 &entry_crop_region_user) == OK) {
524 const uint32_t ir_crop_region[4] = {0, 0, 640, 480};
525 physical_request.settings->Set(
526 ANDROID_SCALER_CROP_REGION,
527 reinterpret_cast<const int32_t*>(&ir_crop_region),
528 sizeof(ir_crop_region) / sizeof(int32_t));
529 }
530 }
531 // Requests for IR pipelines should not include any input buffer or metadata
532 // physical_request.input_buffers
533 // physical_request.input_buffer_metadata
534
535 StreamBuffer internal_buffer = {};
536 status_t res = internal_stream_manager_->GetStreamBuffer(
537 ir_raw_stream_id_[stream_id_index], &internal_buffer);
538 if (res != OK) {
539 ALOGE(
540 "%s: Failed to get internal stream buffer for frame %d, stream id"
541 " %d: %s(%d)",
542 __FUNCTION__, request.frame_number, ir_raw_stream_id_[0],
543 strerror(-res), res);
544 return UNKNOWN_ERROR;
545 }
546 physical_request.output_buffers.push_back(internal_buffer);
547
548 physical_request.physical_camera_settings[camera_id] =
549 HalCameraMetadata::Clone(request.settings.get());
550
551 block_requests->push_back(std::move(block_request));
552
553 return OK;
554 }
555
TryAddRgbProcessBlockRequestLocked(std::vector<ProcessBlockRequest> * block_requests,const CaptureRequest & request)556 status_t RgbirdRtRequestProcessor::TryAddRgbProcessBlockRequestLocked(
557 std::vector<ProcessBlockRequest>* block_requests,
558 const CaptureRequest& request) {
559 ATRACE_CALL();
560 if (block_requests == nullptr) {
561 ALOGE("%s: block_requests is nullptr.", __FUNCTION__);
562 return UNKNOWN_ERROR;
563 }
564
565 ProcessBlockRequest block_request = {.request_id = kRgbCameraId};
566 CaptureRequest& physical_request = block_request.request;
567
568 for (auto& output_buffer : request.output_buffers) {
569 if (output_buffer.stream_id != depth_stream_id_) {
570 physical_request.output_buffers.push_back(output_buffer);
571 }
572 }
573
574 if (is_hdrplus_zsl_enabled_ && request.settings != nullptr) {
575 camera_metadata_ro_entry entry = {};
576 status_t res =
577 request.settings->Get(VendorTagIds::kThermalThrottling, &entry);
578 if (res != OK || entry.count != 1) {
579 ALOGW("%s: Getting thermal throttling entry failed: %s(%d)", __FUNCTION__,
580 strerror(-res), res);
581 } else if (entry.data.u8[0] == true) {
582 // Disable HDR+ once thermal throttles.
583 is_hdrplus_zsl_enabled_ = false;
584 ALOGI("%s: HDR+ ZSL disabled due to thermal throttling", __FUNCTION__);
585 }
586 }
587
588 // Disable HDR+ for thermal throttling.
589 if (is_hdrplus_zsl_enabled_) {
590 status_t res = TryAddHdrplusRawOutputLocked(&physical_request, request);
591 if (res != OK) {
592 ALOGE("%s: AddHdrplusRawOutput fail", __FUNCTION__);
593 return res;
594 }
595 } else if (physical_request.output_buffers.empty() ||
596 IsAutocalRequest(request.frame_number)) {
597 status_t res = TryAddDepthInternalYuvOutputLocked(&physical_request);
598 if (res != OK) {
599 ALOGE("%s: AddDepthOnlyRawOutput failed.", __FUNCTION__);
600 return res;
601 }
602 }
603
604 // In case there is only one depth stream
605 if (!physical_request.output_buffers.empty()) {
606 physical_request.frame_number = request.frame_number;
607 physical_request.settings = HalCameraMetadata::Clone(request.settings.get());
608
609 if (is_hdrplus_zsl_enabled_ && physical_request.settings != nullptr) {
610 status_t res = hal_utils::ModifyRealtimeRequestForHdrplus(
611 physical_request.settings.get());
612 if (res != OK) {
613 ALOGE("%s: ModifyRealtimeRequestForHdrplus (%d) fail", __FUNCTION__,
614 request.frame_number);
615 return UNKNOWN_ERROR;
616 }
617 }
618
619 physical_request.input_buffers = request.input_buffers;
620
621 for (auto& metadata : request.input_buffer_metadata) {
622 physical_request.input_buffer_metadata.push_back(
623 HalCameraMetadata::Clone(metadata.get()));
624 }
625
626 block_requests->push_back(std::move(block_request));
627 }
628 return OK;
629 }
630
TryAddDepthInternalYuvOutputLocked(CaptureRequest * block_request)631 status_t RgbirdRtRequestProcessor::TryAddDepthInternalYuvOutputLocked(
632 CaptureRequest* block_request) {
633 if (block_request == nullptr) {
634 ALOGE("%s: block_request is nullptr.", __FUNCTION__);
635 return UNKNOWN_ERROR;
636 }
637
638 StreamBuffer buffer = {};
639 status_t result =
640 internal_stream_manager_->GetStreamBuffer(rgb_yuv_stream_id_, &buffer);
641 if (result != OK) {
642 ALOGE("%s: GetStreamBuffer failed.", __FUNCTION__);
643 return UNKNOWN_ERROR;
644 }
645 block_request->output_buffers.push_back(buffer);
646
647 return OK;
648 }
649
TryAddHdrplusRawOutputLocked(CaptureRequest * block_request,const CaptureRequest & request)650 status_t RgbirdRtRequestProcessor::TryAddHdrplusRawOutputLocked(
651 CaptureRequest* block_request, const CaptureRequest& request) {
652 ATRACE_CALL();
653 if (block_request == nullptr) {
654 ALOGE("%s: block_request is nullptr.", __FUNCTION__);
655 return UNKNOWN_ERROR;
656 }
657
658 // Update if preview intent has been requested.
659 camera_metadata_ro_entry entry;
660 if (!preview_intent_seen_ && request.settings != nullptr &&
661 request.settings->Get(ANDROID_CONTROL_CAPTURE_INTENT, &entry) == OK) {
662 if (entry.count == 1 &&
663 *entry.data.u8 == ANDROID_CONTROL_CAPTURE_INTENT_PREVIEW) {
664 preview_intent_seen_ = true;
665 ALOGI("%s: First request with preview intent. ZSL starts.", __FUNCTION__);
666 }
667 }
668
669 // Get one RAW bffer from internal stream manager
670 // Add RAW output to capture request
671 if (preview_intent_seen_) {
672 StreamBuffer buffer = {};
673 status_t result =
674 internal_stream_manager_->GetStreamBuffer(rgb_raw_stream_id_, &buffer);
675 if (result != OK) {
676 ALOGE("%s: frame:%d GetStreamBuffer failed.", __FUNCTION__,
677 request.frame_number);
678 return UNKNOWN_ERROR;
679 }
680 block_request->output_buffers.push_back(buffer);
681 }
682
683 return OK;
684 }
685
ProcessRequest(const CaptureRequest & request)686 status_t RgbirdRtRequestProcessor::ProcessRequest(const CaptureRequest& request) {
687 ATRACE_CALL();
688 std::lock_guard<std::mutex> lock(process_block_lock_);
689 if (process_block_ == nullptr) {
690 ALOGE("%s: Not configured yet.", __FUNCTION__);
691 return NO_INIT;
692 }
693
694 // Rgbird should not have phys settings
695 if (!request.physical_camera_settings.empty()) {
696 ALOGE("%s: Rgbird capture session does not support physical settings.",
697 __FUNCTION__);
698 return UNKNOWN_ERROR;
699 }
700
701 {
702 std::vector<ProcessBlockRequest> block_requests;
703 status_t res = TryAddRgbProcessBlockRequestLocked(&block_requests, request);
704 if (res != OK) {
705 ALOGE("%s: Failed to add process block request for rgb pipeline.",
706 __FUNCTION__);
707 return res;
708 }
709
710 // TODO(b/128633958): Remove the force flag after FLL sync is verified
711 if (force_internal_stream_ || depth_stream_id_ != kStreamIdInvalid) {
712 res = AddIrRawProcessBlockRequestLocked(&block_requests, request,
713 kIr1CameraId);
714 if (res != OK) {
715 ALOGE("%s: Failed to add process block request for ir1 pipeline.",
716 __FUNCTION__);
717 return res;
718 }
719 res = AddIrRawProcessBlockRequestLocked(&block_requests, request,
720 kIr2CameraId);
721 if (res != OK) {
722 ALOGE("%s: Failed to add process block request for ir2 pipeline.",
723 __FUNCTION__);
724 return res;
725 }
726 }
727
728 return process_block_->ProcessRequests(block_requests, request);
729 }
730 }
731
Flush()732 status_t RgbirdRtRequestProcessor::Flush() {
733 ATRACE_CALL();
734 std::lock_guard<std::mutex> lock(process_block_lock_);
735 if (process_block_ == nullptr) {
736 return OK;
737 }
738
739 return process_block_->Flush();
740 }
741
742 } // namespace google_camera_hal
743 } // namespace android