[libcamera-devel] [RFC PATCH 6/6] android: camera_device: Support stream mapping
Hirokazu Honda
hiroh at chromium.org
Thu Feb 4 11:05:41 CET 2021
This add the support of stream mapping in Android HAL adaptation
layer. It helps to produce a stream that cannot be output by
a native camera directly.
Signed-off-by: Hirokazu Honda <hiroh at chromium.org>
---
src/android/camera_device.cpp | 143 ++++++++++++++++++++++++++++++++++
1 file changed, 143 insertions(+)
diff --git a/src/android/camera_device.cpp b/src/android/camera_device.cpp
index a1ef07b6..d45f83c1 100644
--- a/src/android/camera_device.cpp
+++ b/src/android/camera_device.cpp
@@ -316,6 +316,132 @@ void sortCamera3StreamConfigs(std::vector<Camera3StreamConfig> &unsortedConfigs,
unsortedConfigs = sortedConfigs;
}
+bool canConvert(const StreamConfiguration &srcConfig,
+ const camera3_stream_t &dstStream,
+ CameraStream::Type &type,
+ PostProcessorType &postProcessorType)
+{
+ /* NV12 -> JPEG conversion. */
+ if (srcConfig.pixelFormat == formats::NV12 &&
+ dstStream.format == HAL_PIXEL_FORMAT_BLOB &&
+ dstStream.width == srcConfig.size.width &&
+ dstStream.height == srcConfig.size.height) {
+ type = CameraStream::Type::Mapped;
+ postProcessorType = PostProcessorType::Jpeg;
+ return true;
+ }
+
+ /* NV12 -> NV12 */
+ if (srcConfig.pixelFormat == formats::NV12 &&
+ (dstStream.format == HAL_PIXEL_FORMAT_YCbCr_420_888 ||
+ dstStream.format == HAL_PIXEL_FORMAT_IMPLEMENTATION_DEFINED)) {
+ if (dstStream.width == srcConfig.size.width &&
+ dstStream.height == srcConfig.size.height) {
+ /* Duplicated stream. No post processing is required. */
+ type = CameraStream::Type::Direct;
+ postProcessorType = PostProcessorType::None;
+ return true;
+ } else if(dstStream.width < srcConfig.size.width &&
+ dstStream.height < srcConfig.size.height) {
+ /* Down scaling. */
+ type = CameraStream::Type::Mapped;
+ postProcessorType = PostProcessorType::Yuv;
+ return true;
+ }
+ }
+ return false;
+}
+
+bool mapConfig(
+ std::vector<Camera3StreamConfig> &streamConfigs,
+ const std::vector<std::pair<StreamConfiguration,
+ CameraConfiguration::Status>> &tryConfigs)
+{
+ const StreamConfiguration kInvalidStreamConfiguration{};
+ assert(!kInvalidStreamConfiguration.pixelFormat.isValid());
+ std::vector<camera3_stream_t *> dstStreams;
+ for (size_t i = 0; i < tryConfigs.size(); i++) {
+ LOG(HAL, Debug) << "StreamConfiguration #" << i << ": "
+ << tryConfigs[i].first.toString()
+ << ", tryValidate() result ="
+ << static_cast<int>(tryConfigs[i].second);
+ if (tryConfigs[i].second == CameraConfiguration::Valid)
+ continue;
+
+ for (auto &stream : streamConfigs[i].streams)
+ dstStreams.push_back(stream.stream);
+ streamConfigs[i].streams.clear();
+
+ if (tryConfigs[i].second == CameraConfiguration::Adjusted) {
+ streamConfigs[i].config =
+ std::move(tryConfigs[i].first);
+ } else {
+ /* Set invalid StreamConfiguration for a
+ * CameraConfiguration::Invalid stream.
+ */
+ streamConfigs[i].config = kInvalidStreamConfiguration;
+ }
+ }
+
+ /* Remove elements in streamConfigs whose stream is evaluated as
+ * CameraConfiguration::Invalid.
+ */
+ streamConfigs.erase(
+ std::remove_if(streamConfigs.begin(), streamConfigs.end(),
+ [](const auto &cfg) {
+ return !cfg.config.pixelFormat.isValid();
+ }),
+ streamConfigs.end());
+
+ /* Find a new source StreamConfiguration from which |dstStream| can be
+ * produced.
+ */
+ for (auto *const dstStream : dstStreams) {
+ auto bestPostProcessorType = PostProcessorType::None;
+ auto bestType = CameraStream::Type::Mapped;
+ Camera3StreamConfig *bestConfig = nullptr;
+
+ for (auto &streamConfig : streamConfigs) {
+ auto type = CameraStream::Type::Mapped;
+ auto postProcessorType = PostProcessorType::None;
+
+ if (canConvert(streamConfig.config, *dstStream, type,
+ postProcessorType) &&
+ static_cast<uint8_t>(type) < static_cast<uint8_t>(bestType)) {
+ bestType = type;
+ bestPostProcessorType = postProcessorType;
+ bestConfig = &streamConfig;
+ }
+ }
+
+ if (!bestConfig) {
+ LOG(HAL, Error)
+ << "No convertible native stream for the stream"
+ << ", width: " << dstStream->width
+ << ", height: " << dstStream->height
+ << ", format: " << utils::hex(dstStream->format);
+ return false;
+ }
+
+ Camera3StreamConfig::Camera3Stream stream(
+ dstStream, bestType, bestPostProcessorType);
+ LOG(HAL, Error)
+ << " Mapped to StreamConfiguration: "
+ << bestConfig->config.toString()
+ << " from " << stream.toString();
+ bestConfig->streams.push_back(std::move(stream));
+ }
+
+ /* Remove elements in streamConfigs whose streams will not be used.*/
+ streamConfigs.erase(
+ std::remove_if(streamConfigs.begin(), streamConfigs.end(),
+ [](const auto &cfg) {
+ return !cfg.config.pixelFormat.isValid();
+ }),
+ streamConfigs.end());
+ return true;
+}
+
} /* namespace */
MappedCamera3Buffer::MappedCamera3Buffer(const buffer_handle_t camera3buffer,
@@ -1634,6 +1760,23 @@ int CameraDevice::configureStreams(camera3_stream_configuration_t *stream_list)
sortCamera3StreamConfigs(streamConfigs, jpegStream);
+ std::vector<std::pair<StreamConfiguration, CameraConfiguration::Status>>
+ tryConfigs(streamConfigs.size());
+ for (size_t i = 0; i < streamConfigs.size(); ++i) {
+ LOG(HAL, Debug) << "Camera3StreamConfig #" << i << ": "
+ << streamConfigs[i].toString();
+ tryConfigs[i].first = streamConfigs[i].config;
+ tryConfigs[i].second = CameraConfiguration::Valid;
+ }
+ if (!config_->tryValidate(tryConfigs)) {
+ LOG(HAL, Error) << "Camera configuration invalid";
+ return -EINVAL;
+ }
+ if (!mapConfig(streamConfigs, tryConfigs)) {
+ LOG(HAL, Error) << "Failed finding a stream mapping";
+ return -EINVAL;
+ }
+
std::vector<PostProcessorType> postProcessorTypes;
for (const auto &streamConfig : streamConfigs) {
config_->addConfiguration(streamConfig.config);
--
2.30.0.365.g02bc693789-goog
More information about the libcamera-devel
mailing list