| /* |
| * Copyright (C) 2016 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 "MetadataReader" |
| |
| #include "metadata_reader.h" |
| |
| #include <cutils/log.h> |
| #include <system/camera.h> |
| |
| #include "metadata_common.h" |
| |
| namespace default_camera_hal { |
| |
| MetadataReader::MetadataReader( |
| std::unique_ptr<const android::CameraMetadata> metadata) |
| : metadata_(std::move(metadata)) {} |
| |
| MetadataReader::~MetadataReader() {} |
| |
| int MetadataReader::Facing(int* facing) const { |
| uint8_t metadata_facing = 0; |
| int res = v4l2_camera_hal::SingleTagValue( |
| *metadata_, ANDROID_LENS_FACING, &metadata_facing); |
| if (res) { |
| ALOGE("%s: Failed to get facing from static metadata.", __func__); |
| return res; |
| } |
| |
| switch (metadata_facing) { |
| case (ANDROID_LENS_FACING_FRONT): |
| *facing = CAMERA_FACING_FRONT; |
| break; |
| case (ANDROID_LENS_FACING_BACK): |
| *facing = CAMERA_FACING_BACK; |
| break; |
| case (ANDROID_LENS_FACING_EXTERNAL): |
| *facing = CAMERA_FACING_EXTERNAL; |
| break; |
| default: |
| ALOGE("%s: Invalid facing from static metadata: %d.", |
| __func__, |
| metadata_facing); |
| return -EINVAL; |
| } |
| return 0; |
| } |
| |
| int MetadataReader::Orientation(int* orientation) const { |
| int32_t metadata_orientation = 0; |
| int res = v4l2_camera_hal::SingleTagValue( |
| *metadata_, ANDROID_SENSOR_ORIENTATION, &metadata_orientation); |
| if (res) { |
| ALOGE("%s: Failed to get orientation from static metadata.", __func__); |
| return res; |
| } |
| |
| // Orientation must be 0, 90, 180, or 270. |
| if (metadata_orientation < 0 || metadata_orientation > 270 || |
| metadata_orientation % 90 != 0) { |
| ALOGE( |
| "%s: Invalid orientation %d " |
| "(must be a 90-degree increment in [0, 360)).", |
| __func__, |
| metadata_orientation); |
| return -EINVAL; |
| } |
| |
| *orientation = static_cast<int>(metadata_orientation); |
| return 0; |
| } |
| |
| int MetadataReader::MaxInputStreams(int32_t* max_input) const { |
| int res = v4l2_camera_hal::SingleTagValue( |
| *metadata_, ANDROID_REQUEST_MAX_NUM_INPUT_STREAMS, max_input); |
| if (res == -ENOENT) { |
| // Not required; default to 0. |
| *max_input = 0; |
| } else if (res) { |
| ALOGE("%s: Failed to get max output streams from static metadata.", |
| __func__); |
| return res; |
| } |
| return 0; |
| } |
| |
| int MetadataReader::MaxOutputStreams(int32_t* max_raw, |
| int32_t* max_non_stalling, |
| int32_t* max_stalling) const { |
| std::array<int32_t, 3> max_output_streams; |
| int res = v4l2_camera_hal::SingleTagValue( |
| *metadata_, ANDROID_REQUEST_MAX_NUM_OUTPUT_STREAMS, &max_output_streams); |
| if (res) { |
| ALOGE("%s: Failed to get max output streams from static metadata.", |
| __func__); |
| return res; |
| } |
| *max_raw = max_output_streams[0]; |
| *max_non_stalling = max_output_streams[1]; |
| *max_stalling = max_output_streams[2]; |
| return 0; |
| } |
| |
| int MetadataReader::RequestCapabilities(std::set<uint8_t>* capabilities) const { |
| std::vector<uint8_t> raw_capabilities; |
| int res = v4l2_camera_hal::VectorTagValue( |
| *metadata_, ANDROID_REQUEST_AVAILABLE_CAPABILITIES, &raw_capabilities); |
| if (res) { |
| ALOGE("%s: Failed to get request capabilities from static metadata.", |
| __func__); |
| return res; |
| } |
| |
| // Move from vector to set. |
| capabilities->insert(raw_capabilities.begin(), raw_capabilities.end()); |
| return 0; |
| } |
| |
| int MetadataReader::StreamConfigurations( |
| std::vector<StreamConfiguration>* configs) const { |
| std::vector<RawStreamConfiguration> raw_stream_configs; |
| int res = v4l2_camera_hal::VectorTagValue( |
| *metadata_, |
| ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS, |
| &raw_stream_configs); |
| if (res) { |
| ALOGE("%s: Failed to get stream configs from static metadata.", __func__); |
| return res; |
| } |
| |
| // TODO(b/31384253): check for required configs. |
| |
| // Convert from raw. |
| configs->insert( |
| configs->end(), raw_stream_configs.begin(), raw_stream_configs.end()); |
| |
| // Check that all configs are valid. |
| for (const auto& config : *configs) { |
| // Must have positive dimensions. |
| if (config.spec.width < 1 || config.spec.height < 1) { |
| ALOGE("%s: Invalid stream config: non-positive dimensions (%d, %d).", |
| __func__, |
| config.spec.width, |
| config.spec.height); |
| return -EINVAL; |
| } |
| // Must have a known direction enum. |
| switch (config.direction) { |
| case ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS_OUTPUT: |
| case ANDROID_SCALER_AVAILABLE_STREAM_CONFIGURATIONS_INPUT: |
| break; |
| default: |
| ALOGE("%s: Invalid stream config direction: %d.", |
| __func__, |
| config.direction); |
| return -EINVAL; |
| } |
| } |
| return 0; |
| } |
| |
| int MetadataReader::StreamStallDurations( |
| std::vector<StreamStallDuration>* stalls) const { |
| std::vector<RawStreamStallDuration> raw_stream_stall_durations; |
| int res = |
| v4l2_camera_hal::VectorTagValue(*metadata_, |
| ANDROID_SCALER_AVAILABLE_STALL_DURATIONS, |
| &raw_stream_stall_durations); |
| if (res) { |
| ALOGE("%s: Failed to get stall durations from static metadata.", __func__); |
| return res; |
| } |
| |
| // Convert from raw. |
| stalls->insert(stalls->end(), |
| raw_stream_stall_durations.begin(), |
| raw_stream_stall_durations.end()); |
| // Check that all stalls are valid. |
| for (const auto& stall : *stalls) { |
| // Must have positive dimensions. |
| if (stall.spec.width < 1 || stall.spec.height < 1) { |
| ALOGE("%s: Invalid stall duration: non-positive dimensions (%d, %d).", |
| __func__, |
| stall.spec.width, |
| stall.spec.height); |
| return -EINVAL; |
| } |
| // Must have a non-negative stall. |
| if (stall.duration < 0) { |
| ALOGE("%s: Invalid stall duration: negative stall %lld.", |
| __func__, |
| static_cast<long long>(stall.duration)); |
| return -EINVAL; |
| } |
| // TODO(b/31384253): YUV_420_888, RAW10, RAW12, RAW_OPAQUE, |
| // and IMPLEMENTATION_DEFINED must have 0 stall duration. |
| } |
| |
| return 0; |
| } |
| |
| int MetadataReader::ReprocessFormats(ReprocessFormatMap* reprocess_map) const { |
| std::vector<int32_t> input_output_formats; |
| int res = v4l2_camera_hal::VectorTagValue( |
| *metadata_, |
| ANDROID_SCALER_AVAILABLE_INPUT_OUTPUT_FORMATS_MAP, |
| &input_output_formats); |
| if (res) { |
| ALOGE("%s: Failed to get input output format map from static metadata.", |
| __func__); |
| return res; |
| } |
| |
| // Convert from the raw vector. |
| for (size_t i = 0; i < input_output_formats.size();) { |
| // The map is represented as variable-length entries of the format |
| // input, num_outputs, <outputs>. |
| |
| // Get the input format. |
| int32_t input_format = input_output_formats[i++]; |
| |
| // Find the output begin and end for this format. |
| int32_t num_output_formats = input_output_formats[i++]; |
| if (num_output_formats < 1) { |
| ALOGE( |
| "%s: No output formats for input format %d.", __func__, input_format); |
| return -EINVAL; |
| } |
| size_t outputs_end = i + num_output_formats; |
| if (outputs_end > input_output_formats.size()) { |
| ALOGE("%s: Input format %d requests more data than available.", |
| __func__, |
| input_format); |
| return -EINVAL; |
| } |
| |
| // Copy all the output formats into the map. |
| (*reprocess_map)[input_format].insert( |
| input_output_formats.data() + i, |
| input_output_formats.data() + outputs_end); |
| |
| // Move on to the next entry. |
| i = outputs_end; |
| } |
| |
| // TODO(b/31384253): check for required mappings. |
| |
| return 0; |
| } |
| |
| } // namespace default_camera_hal |