/*
* 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