/* * 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 #include #include #include #include #include #include #include #include #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 std::vector convertTo(const std::vector& from) { std::vector to; to.reserve(from.size()); std::transform(from.begin(), from.end(), std::back_inserter(to), [](const From& x) { return static_cast(x); }); return to; } template std::vector asVectorOf(const From from) { return std::vector({static_cast(from)}); } } // namespace MetadataBuilder& MetadataBuilder::setSupportedHardwareLevel( const camera_metadata_enum_android_info_supported_hardware_level_t hwLevel) { mEntryMap[ANDROID_INFO_SUPPORTED_HARDWARE_LEVEL] = asVectorOf(hwLevel); return *this; } MetadataBuilder& MetadataBuilder::setDeviceId(int32_t deviceId) { mEntryMap[ANDROID_INFO_DEVICE_ID] = std::vector({deviceId}); 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(metadataVal); return *this; } MetadataBuilder& MetadataBuilder::setFlashState( const camera_metadata_enum_android_flash_state_t flashState) { mEntryMap[ANDROID_FLASH_STATE] = asVectorOf(flashState); return *this; } MetadataBuilder& MetadataBuilder::setFlashMode( const camera_metadata_enum_android_flash_mode_t flashMode) { mEntryMap[ANDROID_FLASH_MODE] = asVectorOf(flashMode); return *this; } MetadataBuilder& MetadataBuilder::setLensFacing( const camera_metadata_enum_android_lens_facing lensFacing) { mEntryMap[ANDROID_LENS_FACING] = asVectorOf(lensFacing); return *this; } MetadataBuilder& MetadataBuilder::setSensorReadoutTimestamp( const camera_metadata_enum_android_sensor_readout_timestamp_t sensorReadoutTimestamp) { mEntryMap[ANDROID_SENSOR_READOUT_TIMESTAMP] = asVectorOf(sensorReadoutTimestamp); return *this; } MetadataBuilder& MetadataBuilder::setAvailableFocalLengths( const std::vector& focalLengths) { mEntryMap[ANDROID_LENS_INFO_AVAILABLE_FOCAL_LENGTHS] = focalLengths; return *this; } MetadataBuilder& MetadataBuilder::setFocalLength(float focalLength) { mEntryMap[ANDROID_LENS_FOCAL_LENGTH] = asVectorOf(focalLength); return *this; } MetadataBuilder& MetadataBuilder::setSensorOrientation(int32_t sensorOrientation) { mEntryMap[ANDROID_SENSOR_ORIENTATION] = asVectorOf(sensorOrientation); return *this; } MetadataBuilder& MetadataBuilder::setSensorTimestampSource( const camera_metadata_enum_android_sensor_info_timestamp_source_t timestampSource) { mEntryMap[ANDROID_SENSOR_INFO_TIMESTAMP_SOURCE] = asVectorOf(timestampSource); return *this; } MetadataBuilder& MetadataBuilder::setSensorTimestamp( std::chrono::nanoseconds timestamp) { mEntryMap[ANDROID_SENSOR_TIMESTAMP] = asVectorOf(timestamp.count()); return *this; } MetadataBuilder& MetadataBuilder::setAvailableFaceDetectModes( const std::vector& faceDetectModes) { mEntryMap[ANDROID_STATISTICS_INFO_AVAILABLE_FACE_DETECT_MODES] = convertTo(faceDetectModes); return *this; } MetadataBuilder& MetadataBuilder::setAvailableTestPatternModes( const std::vector& testPatternModes) { mEntryMap[ANDROID_SENSOR_AVAILABLE_TEST_PATTERN_MODES] = convertTo(testPatternModes); return *this; } MetadataBuilder& MetadataBuilder::setAvailableStreamUseCases( const std::vector< camera_metadata_enum_android_scaler_available_stream_use_cases>& availableUseCases) { mEntryMap[ANDROID_SCALER_AVAILABLE_STREAM_USE_CASES] = convertTo(availableUseCases); return *this; } MetadataBuilder& MetadataBuilder::setFaceDetectMode( const camera_metadata_enum_android_statistics_face_detect_mode_t faceDetectMode) { mEntryMap[ANDROID_STATISTICS_FACE_DETECT_MODE] = asVectorOf(faceDetectMode); return *this; } MetadataBuilder& MetadataBuilder::setControlAvailableModes( const std::vector& availableModes) { mEntryMap[ANDROID_CONTROL_AVAILABLE_MODES] = convertTo(availableModes); return *this; } MetadataBuilder& MetadataBuilder::setControlMode( const camera_metadata_enum_android_control_mode_t mode) { mEntryMap[ANDROID_CONTROL_MODE] = asVectorOf(mode); return *this; } MetadataBuilder& MetadataBuilder::setControlAvailableSceneModes( const std::vector& availableSceneModes) { mEntryMap[ANDROID_CONTROL_AVAILABLE_SCENE_MODES] = convertTo(availableSceneModes); return *this; } MetadataBuilder& MetadataBuilder::setControlSceneMode( const camera_metadata_enum_android_control_scene_mode sceneMode) { mEntryMap[ANDROID_CONTROL_SCENE_MODE] = asVectorOf(sceneMode); return *this; } MetadataBuilder& MetadataBuilder::setControlAvailableEffects( const std::vector& availableEffects) { mEntryMap[ANDROID_CONTROL_AVAILABLE_EFFECTS] = convertTo(availableEffects); return *this; } MetadataBuilder& MetadataBuilder::setControlEffectMode( const camera_metadata_enum_android_control_effect_mode_t effectMode) { mEntryMap[ANDROID_CONTROL_EFFECT_MODE] = asVectorOf(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(videoStabilizationModes); return *this; } MetadataBuilder& MetadataBuilder::setControlVideoStabilizationMode( const camera_metadata_enum_android_control_video_stabilization_mode stabilizationMode) { mEntryMap[ANDROID_CONTROL_VIDEO_STABILIZATION_MODE] = asVectorOf(stabilizationMode); return *this; } MetadataBuilder& MetadataBuilder::setControlAfAvailableModes( const std::vector& availableModes) { mEntryMap[ANDROID_CONTROL_AF_AVAILABLE_MODES] = convertTo(availableModes); return *this; } MetadataBuilder& MetadataBuilder::setControlAfMode( const camera_metadata_enum_android_control_af_mode_t mode) { mEntryMap[ANDROID_CONTROL_AF_MODE] = asVectorOf(mode); return *this; } MetadataBuilder& MetadataBuilder::setControlAfState( const camera_metadata_enum_android_control_af_state afState) { mEntryMap[ANDROID_CONTROL_AF_STATE] = asVectorOf(afState); 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(trigger); return *this; } MetadataBuilder& MetadataBuilder::setControlAeAvailableFpsRanges( const std::vector& fpsRanges) { std::vector 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 FpsRange fpsRange) { mEntryMap[ANDROID_CONTROL_AE_TARGET_FPS_RANGE] = std::vector({fpsRange.minFps, fpsRange.maxFps}); return *this; } MetadataBuilder& MetadataBuilder::setControlAeMode( const camera_metadata_enum_android_control_ae_mode_t mode) { mEntryMap[ANDROID_CONTROL_AE_MODE] = asVectorOf(mode); return *this; } MetadataBuilder& MetadataBuilder::setControlAeAvailableModes( const std::vector& modes) { mEntryMap[ANDROID_CONTROL_AE_AVAILABLE_MODES] = convertTo(modes); return *this; } MetadataBuilder& MetadataBuilder::setControlAePrecaptureTrigger( const camera_metadata_enum_android_control_ae_precapture_trigger_t trigger) { mEntryMap[ANDROID_CONTROL_AE_PRECAPTURE_TRIGGER] = asVectorOf(trigger); return *this; } MetadataBuilder& MetadataBuilder::setControlMaxRegions(int32_t maxAeRegions, int32_t maxAwbRegions, int32_t maxAfRegions) { mEntryMap[ANDROID_CONTROL_MAX_REGIONS] = std::vector({maxAeRegions, maxAwbRegions, maxAfRegions}); return *this; } MetadataBuilder& MetadataBuilder::setControlAvailableAwbModes( const std::vector& awbModes) { mEntryMap[ANDROID_CONTROL_AWB_AVAILABLE_MODES] = convertTo(awbModes); return *this; } MetadataBuilder& MetadataBuilder::setControlAwbMode( const camera_metadata_enum_android_control_awb_mode awbMode) { mEntryMap[ANDROID_CONTROL_AWB_MODE] = asVectorOf(awbMode); return *this; } MetadataBuilder& MetadataBuilder::setControlAwbState( const camera_metadata_enum_android_control_awb_state awbState) { mEntryMap[ANDROID_CONTROL_AWB_STATE] = asVectorOf(awbState); 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({lockAvailable}); return *this; } MetadataBuilder& MetadataBuilder::setControlAwbLock( const camera_metadata_enum_android_control_awb_lock awbLock) { mEntryMap[ANDROID_CONTROL_AWB_LOCK] = asVectorOf(awbLock); return *this; } MetadataBuilder& MetadataBuilder::setControlAeAvailableAntibandingModes( const std::vector& antibandingModes) { mEntryMap[ANDROID_CONTROL_AE_AVAILABLE_ANTIBANDING_MODES] = convertTo(antibandingModes); return *this; } MetadataBuilder& MetadataBuilder::setControlAeAntibandingMode( const camera_metadata_enum_android_control_ae_antibanding_mode_t antibandingMode) { mEntryMap[ANDROID_CONTROL_AE_ANTIBANDING_MODE] = asVectorOf(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(lockAvailable); return *this; } MetadataBuilder& MetadataBuilder::setControlAeLock( const camera_metadata_enum_android_control_ae_lock aeLock) { mEntryMap[ANDROID_CONTROL_AE_LOCK] = asVectorOf(aeLock); return *this; } MetadataBuilder& MetadataBuilder::setControlAeRegions( const std::vector& aeRegions) { std::vector 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& afRegions) { std::vector 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& awbRegions) { std::vector 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(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({x, y, width, height}); return *this; } MetadataBuilder& MetadataBuilder::setMaxJpegSize(const int32_t size) { mEntryMap[ANDROID_JPEG_MAX_SIZE] = asVectorOf(size); return *this; } MetadataBuilder& MetadataBuilder::setMaxFrameDuration( const std::chrono::nanoseconds duration) { mEntryMap[ANDROID_SENSOR_INFO_MAX_FRAME_DURATION] = asVectorOf(duration.count()); return *this; } MetadataBuilder& MetadataBuilder::setJpegAvailableThumbnailSizes( const std::vector& thumbnailSizes) { std::vector 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::setJpegGpsCoordinates( const GpsCoordinates& gpsCoordinates) { mEntryMap[ANDROID_JPEG_GPS_COORDINATES] = std::vector({gpsCoordinates.latitude, gpsCoordinates.longitude, gpsCoordinates.altitude}); if (!gpsCoordinates.provider.empty()) { mEntryMap[ANDROID_JPEG_GPS_PROCESSING_METHOD] = std::vector{ gpsCoordinates.provider.begin(), gpsCoordinates.provider.end()}; } if (gpsCoordinates.timestamp.has_value()) { mEntryMap[ANDROID_JPEG_GPS_TIMESTAMP] = asVectorOf(gpsCoordinates.timestamp.value()); } return *this; } MetadataBuilder& MetadataBuilder::setJpegOrientation(const int32_t orientation) { mEntryMap[ANDROID_JPEG_ORIENTATION] = asVectorOf(orientation); return *this; } MetadataBuilder& MetadataBuilder::setJpegQuality(const uint8_t quality) { mEntryMap[ANDROID_JPEG_QUALITY] = asVectorOf(quality); return *this; } MetadataBuilder& MetadataBuilder::setJpegThumbnailSize(const int width, const int height) { mEntryMap[ANDROID_JPEG_THUMBNAIL_SIZE] = std::vector({width, height}); return *this; } MetadataBuilder& MetadataBuilder::setJpegThumbnailQuality(const uint8_t quality) { mEntryMap[ANDROID_JPEG_THUMBNAIL_QUALITY] = asVectorOf(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( {maxRawStreams, maxProcessedStreams, maxStallStreams}); return *this; } MetadataBuilder& MetadataBuilder::setSyncMaxLatency( const camera_metadata_enum_android_sync_max_latency latency) { mEntryMap[ANDROID_SYNC_MAX_LATENCY] = asVectorOf(latency); return *this; } MetadataBuilder& MetadataBuilder::setPipelineMaxDepth(const uint8_t maxDepth) { mEntryMap[ANDROID_REQUEST_PIPELINE_MAX_DEPTH] = asVectorOf(maxDepth); return *this; } MetadataBuilder& MetadataBuilder::setPipelineDepth(const uint8_t depth) { mEntryMap[ANDROID_REQUEST_PIPELINE_DEPTH] = asVectorOf(depth); return *this; } MetadataBuilder& MetadataBuilder::setAvailableRequestCapabilities( const std::vector& requestCapabilities) { mEntryMap[ANDROID_REQUEST_AVAILABLE_CAPABILITIES] = convertTo(requestCapabilities); return *this; } MetadataBuilder& MetadataBuilder::setAvailableOutputStreamConfigurations( const std::vector& streamConfigurations) { std::vector metadataStreamConfigs; std::vector metadataMinFrameDurations; std::vector 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& aberrationCorectionModes) { mEntryMap[ANDROID_COLOR_CORRECTION_AVAILABLE_ABERRATION_MODES] = convertTo(aberrationCorectionModes); return *this; } MetadataBuilder& MetadataBuilder::setAberrationCorrectionMode( const camera_metadata_enum_android_color_correction_aberration_mode aberrationCorrectionMode) { mEntryMap[ANDROID_COLOR_CORRECTION_ABERRATION_MODE] = asVectorOf(aberrationCorrectionMode); return *this; } MetadataBuilder& MetadataBuilder::setAvailableNoiseReductionModes( const std::vector& noiseReductionModes) { mEntryMap[ANDROID_NOISE_REDUCTION_AVAILABLE_NOISE_REDUCTION_MODES] = convertTo(noiseReductionModes); return *this; } MetadataBuilder& MetadataBuilder::setNoiseReductionMode( const camera_metadata_enum_android_noise_reduction_mode noiseReductionMode) { mEntryMap[ANDROID_NOISE_REDUCTION_MODE] = asVectorOf(noiseReductionMode); return *this; } MetadataBuilder& MetadataBuilder::setRequestPartialResultCount( const int partialResultCount) { mEntryMap[ANDROID_REQUEST_PARTIAL_RESULT_COUNT] = asVectorOf(partialResultCount); return *this; } MetadataBuilder& MetadataBuilder::setCroppingType( const camera_metadata_enum_android_scaler_cropping_type croppingType) { mEntryMap[ANDROID_SCALER_CROPPING_TYPE] = asVectorOf(croppingType); return *this; } MetadataBuilder& MetadataBuilder::setMaxFaceCount(const int maxFaceCount) { mEntryMap[ANDROID_STATISTICS_INFO_MAX_FACE_COUNT] = asVectorOf(maxFaceCount); return *this; } MetadataBuilder& MetadataBuilder::setAvailableMaxDigitalZoom(const float maxZoom) { mEntryMap[ANDROID_SCALER_AVAILABLE_MAX_DIGITAL_ZOOM] = asVectorOf(maxZoom); return *this; } MetadataBuilder& MetadataBuilder::setControlZoomRatioRange(const float min, const float max) { mEntryMap[ANDROID_CONTROL_ZOOM_RATIO_RANGE] = std::vector({min, max}); return *this; } MetadataBuilder& MetadataBuilder::setSensorActiveArraySize(int x0, int y0, int x1, int y1) { mEntryMap[ANDROID_SENSOR_INFO_ACTIVE_ARRAY_SIZE] = std::vector({x0, y0, x1, y1}); return *this; } MetadataBuilder& MetadataBuilder::setSensorPixelArraySize(int width, int height) { mEntryMap[ANDROID_SENSOR_INFO_PIXEL_ARRAY_SIZE] = std::vector({width, height}); return *this; } MetadataBuilder& MetadataBuilder::setSensorPhysicalSize(float width, float height) { mEntryMap[ANDROID_SENSOR_INFO_PHYSICAL_SIZE] = std::vector({width, height}); return *this; } MetadataBuilder& MetadataBuilder::setControlAeCompensationRange(int32_t min, int32_t max) { mEntryMap[ANDROID_CONTROL_AE_COMPENSATION_RANGE] = std::vector({min, max}); return *this; } MetadataBuilder& MetadataBuilder::setControlAeCompensationStep( const camera_metadata_rational step) { mEntryMap[ANDROID_CONTROL_AE_COMPENSATION_STEP] = asVectorOf(step); return *this; } MetadataBuilder& MetadataBuilder::setControlAeExposureCompensation( const int32_t exposureCompensation) { mEntryMap[ANDROID_CONTROL_AE_EXPOSURE_COMPENSATION] = asVectorOf(exposureCompensation); return *this; } MetadataBuilder& MetadataBuilder::setControlAeState( const camera_metadata_enum_android_control_ae_state aeState) { mEntryMap[ANDROID_CONTROL_AE_STATE] = asVectorOf(aeState); return *this; } MetadataBuilder& MetadataBuilder::setStatisticsSceneFlicker( const camera_metadata_enum_android_statistics_scene_flicker sceneFlicker) { mEntryMap[ANDROID_STATISTICS_SCENE_FLICKER] = asVectorOf(sceneFlicker); return *this; } MetadataBuilder& MetadataBuilder::setStatisticsHotPixelMapMode( const camera_metadata_enum_android_statistics_hot_pixel_map_mode hotPixelMapMode) { mEntryMap[ANDROID_STATISTICS_HOT_PIXEL_MAP_MODE] = asVectorOf(hotPixelMapMode); return *this; } MetadataBuilder& MetadataBuilder::setStatisticsLensShadingMapMode( const camera_metadata_enum_android_statistics_lens_shading_map_mode lensShadingMapMode) { mEntryMap[ANDROID_STATISTICS_LENS_SHADING_MAP_MODE] = asVectorOf(lensShadingMapMode); return *this; } MetadataBuilder& MetadataBuilder::setLensOpticalStabilizationMode( const camera_metadata_enum_android_lens_optical_stabilization_mode_t opticalStabilizationMode) { mEntryMap[ANDROID_LENS_OPTICAL_STABILIZATION_MODE] = asVectorOf(opticalStabilizationMode); return *this; } MetadataBuilder& MetadataBuilder::setAvailableRequestKeys( const std::vector& keys) { mEntryMap[ANDROID_REQUEST_AVAILABLE_REQUEST_KEYS] = keys; return *this; } MetadataBuilder& MetadataBuilder::setAvailableResultKeys( const std::vector& keys) { mEntryMap[ANDROID_REQUEST_AVAILABLE_RESULT_KEYS] = keys; return *this; } MetadataBuilder& MetadataBuilder::setAvailableCapabilities( const std::vector& capabilities) { mEntryMap[ANDROID_REQUEST_AVAILABLE_CAPABILITIES] = convertTo(capabilities); return *this; } MetadataBuilder& MetadataBuilder::setAvailableCharacteristicKeys( const std::vector& keys) { mEntryMap[ANDROID_REQUEST_AVAILABLE_CHARACTERISTICS_KEYS] = convertTo(keys); return *this; } MetadataBuilder& MetadataBuilder::setAvailableCharacteristicKeys() { mExtendWithAvailableCharacteristicsKeys = true; return *this; } std::unique_ptr MetadataBuilder::build() { if (mExtendWithAvailableCharacteristicsKeys) { std::vector 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(); const uint8_t* data_ptr = reinterpret_cast(metadata); aidlMetadata->metadata.assign(data_ptr, data_ptr + get_camera_metadata_size(metadata)); metadataHelper.unlock(metadata); return aidlMetadata; } std::optional getJpegQuality( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(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; } int32_t getJpegOrientation( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry(metadata, ANDROID_JPEG_ORIENTATION, &entry) != OK) { return 0; } return *entry.data.i32; } std::optional getJpegThumbnailSize( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(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 getJpegThumbnailQuality( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(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 getJpegAvailableThumbnailSizes( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(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 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; } std::optional getFpsRange( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry( metadata, ANDROID_CONTROL_AE_TARGET_FPS_RANGE, &entry) != OK || entry.count != 2) { return {}; } FpsRange range{.minFps = entry.data.i32[0], .maxFps = entry.data.i32[1]}; return range; } std::optional getCaptureIntent(const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry(metadata, ANDROID_CONTROL_CAPTURE_INTENT, &entry) != OK) { return {}; } return static_cast( entry.data.u8[0]); } std::optional getGpsCoordinates( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry(metadata, ANDROID_JPEG_GPS_COORDINATES, &entry) != OK) { return std::nullopt; } GpsCoordinates coordinates{.latitude = entry.data.d[0], .longitude = entry.data.d[1], .altitude = entry.data.d[2]}; if (find_camera_metadata_ro_entry(metadata, ANDROID_JPEG_GPS_TIMESTAMP, &entry) == OK) { coordinates.timestamp = entry.data.i64[0]; } // According to types.hal, the string describing the GPS processing method has // a 32 characters size static constexpr float kGpsProviderStringLength = 32; if (find_camera_metadata_ro_entry( metadata, ANDROID_JPEG_GPS_PROCESSING_METHOD, &entry) == OK) { coordinates.provider.assign( reinterpret_cast(entry.data.u8), std::min(entry.count, static_cast(kGpsProviderStringLength))); } return coordinates; } std::optional getLensFacing( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry(metadata, ANDROID_LENS_FACING, &entry) != OK) { return std::nullopt; } return static_cast(entry.data.u8[0]); } std::optional getPrecaptureTrigger( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry( metadata, ANDROID_CONTROL_AE_PRECAPTURE_TRIGGER, &entry) != OK) { return std::nullopt; } return static_cast( entry.data.u8[0]); } std::optional getDeviceId( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry(metadata, ANDROID_INFO_DEVICE_ID, &entry) != OK) { return std::nullopt; } return static_cast(entry.data.i32[0]); } std::optional getSensorOrientation( const aidl::android::hardware::camera::device::CameraMetadata& cameraMetadata) { auto metadata = reinterpret_cast(cameraMetadata.metadata.data()); camera_metadata_ro_entry_t entry; if (find_camera_metadata_ro_entry(metadata, ANDROID_SENSOR_ORIENTATION, &entry) != OK) { return std::nullopt; } return static_cast(entry.data.i32[0]); } } // namespace virtualcamera } // namespace companion } // namespace android