blob: e3d9e287bbf43812fcad49448922cecf6320c08d [file] [log] [blame]
/*
* Copyright 2023 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// #define LOG_NDEBUG 0
#define LOG_TAG "MetadataUtil"
#include "MetadataUtil.h"
#include <algorithm>
#include <cstdint>
#include <iterator>
#include <memory>
#include <utility>
#include <variant>
#include <vector>
#include "CameraMetadata.h"
#include "aidl/android/hardware/camera/device/CameraMetadata.h"
#include "log/log.h"
#include "system/camera_metadata.h"
#include "util/Util.h"
#include "utils/Errors.h"
namespace android {
namespace companion {
namespace virtualcamera {
namespace {
using ::android::hardware::camera::common::helper::CameraMetadata;
template <typename To, typename From>
std::vector<To> convertTo(const std::vector<From>& from) {
std::vector<To> to;
to.reserve(from.size());
std::transform(from.begin(), from.end(), std::back_inserter(to),
[](const From& x) { return static_cast<To>(x); });
return to;
}
template <typename To, typename From>
std::vector<To> asVectorOf(const From from) {
return std::vector<To>({static_cast<To>(from)});
}
} // namespace
MetadataBuilder& MetadataBuilder::setSupportedHardwareLevel(
camera_metadata_enum_android_info_supported_hardware_level_t hwLevel) {
mEntryMap[ANDROID_INFO_SUPPORTED_HARDWARE_LEVEL] =
asVectorOf<uint8_t>(hwLevel);
return *this;
}
MetadataBuilder& MetadataBuilder::setFlashAvailable(bool flashAvailable) {
const uint8_t metadataVal = flashAvailable
? ANDROID_FLASH_INFO_AVAILABLE_TRUE
: ANDROID_FLASH_INFO_AVAILABLE_FALSE;
mEntryMap[ANDROID_FLASH_INFO_AVAILABLE] = asVectorOf<uint8_t>(metadataVal);
return *this;
}
MetadataBuilder& MetadataBuilder::setFlashState(
const camera_metadata_enum_android_flash_state_t flashState) {
mEntryMap[ANDROID_FLASH_STATE] = asVectorOf<uint8_t>(flashState);
return *this;
}
MetadataBuilder& MetadataBuilder::setFlashMode(
const camera_metadata_enum_android_flash_mode_t flashMode) {
mEntryMap[ANDROID_FLASH_MODE] = asVectorOf<uint8_t>(flashMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setLensFacing(
camera_metadata_enum_android_lens_facing lensFacing) {
mEntryMap[ANDROID_LENS_FACING] = asVectorOf<uint8_t>(lensFacing);
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorReadoutTimestamp(
const camera_metadata_enum_android_sensor_readout_timestamp_t
sensorReadoutTimestamp) {
mEntryMap[ANDROID_SENSOR_READOUT_TIMESTAMP] =
asVectorOf<uint8_t>(sensorReadoutTimestamp);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableFocalLengths(
const std::vector<float>& focalLengths) {
mEntryMap[ANDROID_LENS_INFO_AVAILABLE_FOCAL_LENGTHS] = focalLengths;
return *this;
}
MetadataBuilder& MetadataBuilder::setFocalLength(float focalLength) {
mEntryMap[ANDROID_LENS_FOCAL_LENGTH] = asVectorOf<float>(focalLength);
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorOrientation(int32_t sensorOrientation) {
mEntryMap[ANDROID_SENSOR_ORIENTATION] = asVectorOf<int32_t>(sensorOrientation);
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorTimestampSource(
const camera_metadata_enum_android_sensor_info_timestamp_source_t
timestampSource) {
mEntryMap[ANDROID_SENSOR_INFO_TIMESTAMP_SOURCE] =
asVectorOf<uint8_t>(timestampSource);
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorTimestamp(
std::chrono::nanoseconds timestamp) {
mEntryMap[ANDROID_SENSOR_TIMESTAMP] = asVectorOf<int64_t>(timestamp.count());
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableFaceDetectModes(
const std::vector<camera_metadata_enum_android_statistics_face_detect_mode_t>&
faceDetectModes) {
mEntryMap[ANDROID_STATISTICS_INFO_AVAILABLE_FACE_DETECT_MODES] =
convertTo<uint8_t>(faceDetectModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableTestPatternModes(
const std::vector<camera_metadata_enum_android_sensor_test_pattern_mode>&
testPatternModes) {
mEntryMap[ANDROID_SENSOR_AVAILABLE_TEST_PATTERN_MODES] =
convertTo<int32_t>(testPatternModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setFaceDetectMode(
const camera_metadata_enum_android_statistics_face_detect_mode_t
faceDetectMode) {
mEntryMap[ANDROID_STATISTICS_FACE_DETECT_MODE] =
asVectorOf<uint8_t>(faceDetectMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAvailableModes(
const std::vector<camera_metadata_enum_android_control_mode_t>&
availableModes) {
mEntryMap[ANDROID_CONTROL_AVAILABLE_MODES] =
convertTo<uint8_t>(availableModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlMode(
const camera_metadata_enum_android_control_mode_t mode) {
mEntryMap[ANDROID_CONTROL_MODE] = asVectorOf<uint8_t>(mode);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAvailableSceneModes(
const std::vector<camera_metadata_enum_android_control_scene_mode>&
availableSceneModes) {
mEntryMap[ANDROID_CONTROL_AVAILABLE_SCENE_MODES] =
convertTo<uint8_t>(availableSceneModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAvailableEffects(
const std::vector<camera_metadata_enum_android_control_effect_mode>&
availableEffects) {
mEntryMap[ANDROID_CONTROL_AVAILABLE_EFFECTS] =
convertTo<uint8_t>(availableEffects);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlEffectMode(
const camera_metadata_enum_android_control_effect_mode_t effectMode) {
mEntryMap[ANDROID_CONTROL_EFFECT_MODE] = asVectorOf<uint8_t>(effectMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAvailableVideoStabilizationModes(
const std::vector<
camera_metadata_enum_android_control_video_stabilization_mode_t>&
videoStabilizationModes) {
mEntryMap[ANDROID_CONTROL_AVAILABLE_VIDEO_STABILIZATION_MODES] =
convertTo<uint8_t>(videoStabilizationModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAfAvailableModes(
const std::vector<camera_metadata_enum_android_control_af_mode_t>&
availableModes) {
mEntryMap[ANDROID_CONTROL_AF_AVAILABLE_MODES] =
convertTo<uint8_t>(availableModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAfMode(
const camera_metadata_enum_android_control_af_mode_t mode) {
mEntryMap[ANDROID_CONTROL_AF_MODE] = asVectorOf<uint8_t>(mode);
return *this;
}
// See ANDROID_CONTROL_AF_TRIGGER_MODE in CameraMetadataTag.aidl.
MetadataBuilder& MetadataBuilder::setControlAfTrigger(
const camera_metadata_enum_android_control_af_trigger_t trigger) {
mEntryMap[ANDROID_CONTROL_AF_TRIGGER] = asVectorOf<uint8_t>(trigger);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeAvailableFpsRanges(
const std::vector<FpsRange>& fpsRanges) {
std::vector<int32_t> ranges;
ranges.reserve(2 * fpsRanges.size());
for (const FpsRange fpsRange : fpsRanges) {
ranges.push_back(fpsRange.minFps);
ranges.push_back(fpsRange.maxFps);
}
mEntryMap[ANDROID_CONTROL_AE_AVAILABLE_TARGET_FPS_RANGES] = std::move(ranges);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeTargetFpsRange(
const int32_t minFps, const int32_t maxFps) {
mEntryMap[ANDROID_CONTROL_AE_TARGET_FPS_RANGE] =
std::vector<int32_t>({minFps, maxFps});
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeMode(
camera_metadata_enum_android_control_ae_mode_t mode) {
mEntryMap[ANDROID_CONTROL_AE_MODE] = asVectorOf<uint8_t>(mode);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeAvailableModes(
const std::vector<camera_metadata_enum_android_control_ae_mode_t>& modes) {
mEntryMap[ANDROID_CONTROL_AE_AVAILABLE_MODES] = convertTo<uint8_t>(modes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAePrecaptureTrigger(
const camera_metadata_enum_android_control_ae_precapture_trigger_t trigger) {
mEntryMap[ANDROID_CONTROL_AE_PRECAPTURE_TRIGGER] =
asVectorOf<uint8_t>(trigger);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlMaxRegions(int32_t maxAeRegions,
int32_t maxAwbRegions,
int32_t maxAfRegions) {
mEntryMap[ANDROID_CONTROL_MAX_REGIONS] =
std::vector<int32_t>({maxAeRegions, maxAwbRegions, maxAfRegions});
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAvailableAwbModes(
const std::vector<camera_metadata_enum_android_control_awb_mode>& awbModes) {
mEntryMap[ANDROID_CONTROL_AWB_AVAILABLE_MODES] = convertTo<uint8_t>(awbModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAwbMode(
const camera_metadata_enum_android_control_awb_mode awbMode) {
mEntryMap[ANDROID_CONTROL_AWB_MODE] = asVectorOf<uint8_t>(awbMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAwbLockAvailable(
const bool awbLockAvailable) {
const uint8_t lockAvailable = awbLockAvailable
? ANDROID_CONTROL_AWB_LOCK_AVAILABLE_TRUE
: ANDROID_CONTROL_AWB_LOCK_AVAILABLE_FALSE;
mEntryMap[ANDROID_CONTROL_AWB_LOCK_AVAILABLE] =
std::vector<uint8_t>({lockAvailable});
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeAvailableAntibandingModes(
const std::vector<camera_metadata_enum_android_control_ae_antibanding_mode_t>&
antibandingModes) {
mEntryMap[ANDROID_CONTROL_AE_AVAILABLE_ANTIBANDING_MODES] =
convertTo<uint8_t>(antibandingModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeAntibandingMode(
const camera_metadata_enum_android_control_ae_antibanding_mode_t
antibandingMode) {
mEntryMap[ANDROID_CONTROL_AE_ANTIBANDING_MODE] =
asVectorOf<uint8_t>(antibandingMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeLockAvailable(
const bool aeLockAvailable) {
const uint8_t lockAvailable = aeLockAvailable
? ANDROID_CONTROL_AE_LOCK_AVAILABLE_TRUE
: ANDROID_CONTROL_AE_LOCK_AVAILABLE_FALSE;
mEntryMap[ANDROID_CONTROL_AE_LOCK_AVAILABLE] =
asVectorOf<uint8_t>(lockAvailable);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeRegions(
const std::vector<ControlRegion>& aeRegions) {
std::vector<int32_t> regions;
regions.reserve(5 * aeRegions.size());
for (const ControlRegion& region : aeRegions) {
regions.push_back(region.x0);
regions.push_back(region.y0);
regions.push_back(region.x1);
regions.push_back(region.y1);
regions.push_back(region.weight);
}
mEntryMap[ANDROID_CONTROL_AE_REGIONS] = std::move(regions);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAfRegions(
const std::vector<ControlRegion>& afRegions) {
std::vector<int32_t> regions;
regions.reserve(5 * afRegions.size());
for (const ControlRegion& region : afRegions) {
regions.push_back(region.x0);
regions.push_back(region.y0);
regions.push_back(region.x1);
regions.push_back(region.y1);
regions.push_back(region.weight);
}
mEntryMap[ANDROID_CONTROL_AF_REGIONS] = std::move(regions);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAwbRegions(
const std::vector<ControlRegion>& awbRegions) {
std::vector<int32_t> regions;
regions.reserve(5 * awbRegions.size());
for (const ControlRegion& region : awbRegions) {
regions.push_back(region.x0);
regions.push_back(region.y0);
regions.push_back(region.x1);
regions.push_back(region.y1);
regions.push_back(region.weight);
}
mEntryMap[ANDROID_CONTROL_AWB_REGIONS] = std::move(regions);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlCaptureIntent(
const camera_metadata_enum_android_control_capture_intent_t intent) {
mEntryMap[ANDROID_CONTROL_CAPTURE_INTENT] = asVectorOf<uint8_t>(intent);
return *this;
}
MetadataBuilder& MetadataBuilder::setCropRegion(const int32_t x, const int32_t y,
const int32_t width,
const int32_t height) {
mEntryMap[ANDROID_SCALER_CROP_REGION] =
std::vector<int32_t>({x, y, width, height});
return *this;
}
MetadataBuilder& MetadataBuilder::setMaxJpegSize(const int32_t size) {
mEntryMap[ANDROID_JPEG_MAX_SIZE] = asVectorOf<int32_t>(size);
return *this;
}
MetadataBuilder& MetadataBuilder::setMaxFrameDuration(
const std::chrono::nanoseconds duration) {
mEntryMap[ANDROID_SENSOR_INFO_MAX_FRAME_DURATION] =
asVectorOf<int64_t>(duration.count());
return *this;
}
MetadataBuilder& MetadataBuilder::setJpegAvailableThumbnailSizes(
const std::vector<Resolution>& thumbnailSizes) {
std::vector<int32_t> sizes;
sizes.reserve(thumbnailSizes.size() * 2);
for (const Resolution& resolution : thumbnailSizes) {
sizes.push_back(resolution.width);
sizes.push_back(resolution.height);
}
mEntryMap[ANDROID_JPEG_AVAILABLE_THUMBNAIL_SIZES] = std::move(sizes);
return *this;
}
MetadataBuilder& MetadataBuilder::setJpegQuality(const uint8_t quality) {
mEntryMap[ANDROID_JPEG_QUALITY] = asVectorOf<uint8_t>(quality);
return *this;
}
MetadataBuilder& MetadataBuilder::setJpegThumbnailSize(const int width,
const int height) {
mEntryMap[ANDROID_JPEG_THUMBNAIL_SIZE] = std::vector<int32_t>({width, height});
return *this;
}
MetadataBuilder& MetadataBuilder::setJpegThumbnailQuality(const uint8_t quality) {
mEntryMap[ANDROID_JPEG_THUMBNAIL_QUALITY] = asVectorOf<uint8_t>(quality);
return *this;
}
MetadataBuilder& MetadataBuilder::setMaxNumberOutputStreams(
const int32_t maxRawStreams, const int32_t maxProcessedStreams,
const int32_t maxStallStreams) {
mEntryMap[ANDROID_REQUEST_MAX_NUM_OUTPUT_STREAMS] = std::vector<int32_t>(
{maxRawStreams, maxProcessedStreams, maxStallStreams});
return *this;
}
MetadataBuilder& MetadataBuilder::setSyncMaxLatency(
camera_metadata_enum_android_sync_max_latency latency) {
mEntryMap[ANDROID_SYNC_MAX_LATENCY] = asVectorOf<int32_t>(latency);
return *this;
}
MetadataBuilder& MetadataBuilder::setPipelineMaxDepth(const uint8_t maxDepth) {
mEntryMap[ANDROID_REQUEST_PIPELINE_MAX_DEPTH] = asVectorOf<uint8_t>(maxDepth);
return *this;
}
MetadataBuilder& MetadataBuilder::setPipelineDepth(const uint8_t depth) {
mEntryMap[ANDROID_REQUEST_PIPELINE_DEPTH] = asVectorOf<uint8_t>(depth);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableRequestCapabilities(
const std::vector<camera_metadata_enum_android_request_available_capabilities_t>&
requestCapabilities) {
mEntryMap[ANDROID_REQUEST_AVAILABLE_CAPABILITIES] =
convertTo<uint8_t>(requestCapabilities);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableOutputStreamConfigurations(
const std::vector<StreamConfiguration>& streamConfigurations) {
std::vector<int32_t> metadataStreamConfigs;
std::vector<int64_t> metadataMinFrameDurations;
std::vector<int64_t> metadataStallDurations;
metadataStreamConfigs.reserve(streamConfigurations.size());
metadataMinFrameDurations.reserve(streamConfigurations.size());
metadataStallDurations.reserve(streamConfigurations.size());
for (const auto& config : streamConfigurations) {
metadataStreamConfigs.push_back(config.format);
metadataStreamConfigs.push_back(config.width);
metadataStreamConfigs.push_back(config.height);
metadataStreamConfigs.push_back(
ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS_OUTPUT);
metadataMinFrameDurations.push_back(config.format);
metadataMinFrameDurations.push_back(config.width);
metadataMinFrameDurations.push_back(config.height);
metadataMinFrameDurations.push_back(config.minFrameDuration.count());
metadataStallDurations.push_back(config.format);
metadataStallDurations.push_back(config.width);
metadataStallDurations.push_back(config.height);
metadataStallDurations.push_back(config.minStallDuration.count());
}
mEntryMap[ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS] =
std::move(metadataStreamConfigs);
mEntryMap[ANDROID_SCALER_AVAILABLE_MIN_FRAME_DURATIONS] =
std::move(metadataMinFrameDurations);
mEntryMap[ANDROID_SCALER_AVAILABLE_STALL_DURATIONS] =
std::move(metadataStallDurations);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableAberrationCorrectionModes(
const std::vector<camera_metadata_enum_android_color_correction_aberration_mode>&
aberrationCorectionModes) {
mEntryMap[ANDROID_COLOR_CORRECTION_AVAILABLE_ABERRATION_MODES] =
convertTo<uint8_t>(aberrationCorectionModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setAberrationCorrectionMode(
const camera_metadata_enum_android_color_correction_aberration_mode
aberrationCorrectionMode) {
mEntryMap[ANDROID_COLOR_CORRECTION_ABERRATION_MODE] =
asVectorOf<uint8_t>(aberrationCorrectionMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableNoiseReductionModes(
const std::vector<camera_metadata_enum_android_noise_reduction_mode>&
noiseReductionModes) {
mEntryMap[ANDROID_NOISE_REDUCTION_AVAILABLE_NOISE_REDUCTION_MODES] =
convertTo<uint8_t>(noiseReductionModes);
return *this;
}
MetadataBuilder& MetadataBuilder::setNoiseReductionMode(
camera_metadata_enum_android_noise_reduction_mode noiseReductionMode) {
mEntryMap[ANDROID_NOISE_REDUCTION_MODE] =
asVectorOf<uint8_t>(noiseReductionMode);
return *this;
}
MetadataBuilder& MetadataBuilder::setRequestPartialResultCount(
const int partialResultCount) {
mEntryMap[ANDROID_REQUEST_PARTIAL_RESULT_COUNT] =
asVectorOf<int32_t>(partialResultCount);
return *this;
}
MetadataBuilder& MetadataBuilder::setCroppingType(
const camera_metadata_enum_android_scaler_cropping_type croppingType) {
mEntryMap[ANDROID_SCALER_CROPPING_TYPE] = asVectorOf<uint8_t>(croppingType);
return *this;
}
MetadataBuilder& MetadataBuilder::setMaxFaceCount(const int maxFaceCount) {
mEntryMap[ANDROID_STATISTICS_INFO_MAX_FACE_COUNT] =
asVectorOf<int32_t>(maxFaceCount);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableMaxDigitalZoom(const float maxZoom) {
mEntryMap[ANDROID_SCALER_AVAILABLE_MAX_DIGITAL_ZOOM] =
asVectorOf<float>(maxZoom);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlZoomRatioRange(const float min,
const float max) {
mEntryMap[ANDROID_CONTROL_ZOOM_RATIO_RANGE] = std::vector<float>({min, max});
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorActiveArraySize(int x0, int y0,
int x1, int y1) {
mEntryMap[ANDROID_SENSOR_INFO_ACTIVE_ARRAY_SIZE] =
std::vector<int32_t>({x0, y0, x1, y1});
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorPixelArraySize(int width,
int height) {
mEntryMap[ANDROID_SENSOR_INFO_PIXEL_ARRAY_SIZE] =
std::vector<int32_t>({width, height});
return *this;
}
MetadataBuilder& MetadataBuilder::setSensorPhysicalSize(float width,
float height) {
mEntryMap[ANDROID_SENSOR_INFO_PHYSICAL_SIZE] =
std::vector<float>({width, height});
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeCompensationRange(int32_t min,
int32_t max) {
mEntryMap[ANDROID_CONTROL_AE_COMPENSATION_RANGE] =
std::vector<int32_t>({min, max});
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeCompensationStep(
const camera_metadata_rational step) {
mEntryMap[ANDROID_CONTROL_AE_COMPENSATION_STEP] =
asVectorOf<camera_metadata_rational>(step);
return *this;
}
MetadataBuilder& MetadataBuilder::setControlAeExposureCompensation(
const int32_t exposureCompensation) {
mEntryMap[ANDROID_CONTROL_AE_EXPOSURE_COMPENSATION] =
asVectorOf<int32_t>(exposureCompensation);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableRequestKeys(
const std::vector<int32_t>& keys) {
mEntryMap[ANDROID_REQUEST_AVAILABLE_REQUEST_KEYS] = keys;
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableResultKeys(
const std::vector<int32_t>& keys) {
mEntryMap[ANDROID_REQUEST_AVAILABLE_RESULT_KEYS] = keys;
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableCapabilities(
const std::vector<camera_metadata_enum_android_request_available_capabilities_t>&
capabilities) {
mEntryMap[ANDROID_REQUEST_AVAILABLE_CAPABILITIES] =
convertTo<uint8_t>(capabilities);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableCharacteristicKeys(
const std::vector<camera_metadata_tag_t>& keys) {
mEntryMap[ANDROID_REQUEST_AVAILABLE_CHARACTERISTICS_KEYS] =
convertTo<int32_t>(keys);
return *this;
}
MetadataBuilder& MetadataBuilder::setAvailableCharacteristicKeys() {
mExtendWithAvailableCharacteristicsKeys = true;
return *this;
}
std::unique_ptr<aidl::android::hardware::camera::device::CameraMetadata>
MetadataBuilder::build() {
if (mExtendWithAvailableCharacteristicsKeys) {
std::vector<camera_metadata_tag_t> availableKeys;
availableKeys.reserve(mEntryMap.size());
for (const auto& [key, _] : mEntryMap) {
if (key != ANDROID_REQUEST_AVAILABLE_CHARACTERISTICS_KEYS) {
availableKeys.push_back(key);
}
}
setAvailableCharacteristicKeys(availableKeys);
}
CameraMetadata metadataHelper;
for (const auto& entry : mEntryMap) {
status_t ret = std::visit(
[&](auto&& arg) {
return metadataHelper.update(entry.first, arg.data(), arg.size());
},
entry.second);
if (ret != NO_ERROR) {
ALOGE("Failed to update metadata with key %d - %s: %s", entry.first,
get_camera_metadata_tag_name(entry.first),
::android::statusToString(ret).c_str());
return nullptr;
}
}
const camera_metadata_t* metadata = metadataHelper.getAndLock();
if (metadata == nullptr) {
ALOGE(
"Failure when constructing metadata -> CameraMetadata helper returned "
"nullptr");
return nullptr;
}
auto aidlMetadata =
std::make_unique<aidl::android::hardware::camera::device::CameraMetadata>();
const uint8_t* data_ptr = reinterpret_cast<const uint8_t*>(metadata);
aidlMetadata->metadata.assign(data_ptr,
data_ptr + get_camera_metadata_size(metadata));
metadataHelper.unlock(metadata);
return aidlMetadata;
}
std::optional<int32_t> getJpegQuality(
const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) {
auto metadata =
reinterpret_cast<const camera_metadata_t*>(cameraMetadata.metadata.data());
camera_metadata_ro_entry_t entry;
if (find_camera_metadata_ro_entry(metadata, ANDROID_JPEG_QUALITY, &entry) !=
OK) {
return std::nullopt;
}
return *entry.data.i32;
}
std::optional<Resolution> getJpegThumbnailSize(
const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) {
auto metadata =
reinterpret_cast<const camera_metadata_t*>(cameraMetadata.metadata.data());
camera_metadata_ro_entry_t entry;
if (find_camera_metadata_ro_entry(metadata, ANDROID_JPEG_THUMBNAIL_SIZE,
&entry) != OK) {
return std::nullopt;
}
return Resolution(entry.data.i32[0], entry.data.i32[1]);
}
std::optional<int32_t> getJpegThumbnailQuality(
const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) {
auto metadata =
reinterpret_cast<const camera_metadata_t*>(cameraMetadata.metadata.data());
camera_metadata_ro_entry_t entry;
if (find_camera_metadata_ro_entry(metadata, ANDROID_JPEG_THUMBNAIL_QUALITY,
&entry) != OK) {
return std::nullopt;
}
return *entry.data.i32;
}
std::vector<Resolution> getJpegAvailableThumbnailSizes(
const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) {
auto metadata =
reinterpret_cast<const camera_metadata_t*>(cameraMetadata.metadata.data());
camera_metadata_ro_entry_t entry;
if (find_camera_metadata_ro_entry(
metadata, ANDROID_JPEG_AVAILABLE_THUMBNAIL_SIZES, &entry) != OK) {
return {};
}
std::vector<Resolution> thumbnailSizes;
thumbnailSizes.reserve(entry.count / 2);
for (int i = 0; i < entry.count; i += 2) {
thumbnailSizes.emplace_back(entry.data.i32[i], entry.data.i32[i + 1]);
}
return thumbnailSizes;
}
} // namespace virtualcamera
} // namespace companion
} // namespace android