From patchwork Thu Aug 26 21:30:12 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Umang Jain X-Patchwork-Id: 13516 X-Patchwork-Delegate: umang.jain@ideasonboard.com Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id 4167FC3241 for ; Thu, 26 Aug 2021 21:30:30 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id D2AEA68928; Thu, 26 Aug 2021 23:30:29 +0200 (CEST) Authentication-Results: lancelot.ideasonboard.com; dkim=fail reason="signature verification failed" (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="GuWIhXOp"; dkim-atps=neutral Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [IPv6:2001:4b98:dc2:55:216:3eff:fef7:d647]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id 4EA1D68939 for ; Thu, 26 Aug 2021 23:30:28 +0200 (CEST) Received: from perceval.ideasonboard.com (unknown [103.251.226.246]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id 2E304191F; Thu, 26 Aug 2021 23:30:26 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1630013428; bh=gJ+LbJJ+dxzljpBCnlUKcnMNM/xGWD/NL5UWZu7Y4bo=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=GuWIhXOp6KevLY7ygSc3JEfVmxhuYjP/1YeKvI0grfcHAKLlRVPUTNi/DbbFEWRI/ KKw1mMBpJQnCNa2xpnbabekRZz/g5oLYBLK86pKdYwKMVLOdim/Fud6CH53GiuT/Z/ mKLbj3OsOwIMmTOWY/5OtE4zJhpwixHOP4OGc4V4= From: Umang Jain To: libcamera-devel@lists.libcamera.org Date: Fri, 27 Aug 2021 03:00:12 +0530 Message-Id: <20210826213016.1829504-2-umang.jain@ideasonboard.com> X-Mailer: git-send-email 2.31.1 In-Reply-To: <20210826213016.1829504-1-umang.jain@ideasonboard.com> References: <20210826213016.1829504-1-umang.jain@ideasonboard.com> MIME-Version: 1.0 Subject: [libcamera-devel] [RFC PATCH 1/5] android: post_processor: Inherit from libcamera::Object X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" Currently, the post processing of a stream happens synchronously in CameraStream::process(). There is a plan to offload this processing to a separate worker thread. In order to achieve that, we need to move the post processor to a separate thread and invoke methods on it from the CameraStream's thread. Object::moveToThread() and Object::invokeMethod() shall be used respectively, to achieve that goal. Hence, inherit the PostProcessor interface from libcamera::object. Signed-off-by: Umang Jain --- src/android/post_processor.h | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/android/post_processor.h b/src/android/post_processor.h index ab2b2c60..b2b5bd55 100644 --- a/src/android/post_processor.h +++ b/src/android/post_processor.h @@ -7,6 +7,8 @@ #ifndef __ANDROID_POST_PROCESSOR_H__ #define __ANDROID_POST_PROCESSOR_H__ +#include + #include #include @@ -14,7 +16,7 @@ class CameraMetadata; -class PostProcessor +class PostProcessor : public libcamera::Object { public: virtual ~PostProcessor() = default; From patchwork Thu Aug 26 21:30:13 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Umang Jain X-Patchwork-Id: 13517 X-Patchwork-Delegate: umang.jain@ideasonboard.com Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id CC5E7BD87D for ; Thu, 26 Aug 2021 21:30:32 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id 8347468932; Thu, 26 Aug 2021 23:30:32 +0200 (CEST) Authentication-Results: lancelot.ideasonboard.com; dkim=fail reason="signature verification failed" (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="H0pr7Phj"; dkim-atps=neutral Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [IPv6:2001:4b98:dc2:55:216:3eff:fef7:d647]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id 43F6B6893C for ; Thu, 26 Aug 2021 23:30:30 +0200 (CEST) Received: from perceval.ideasonboard.com (unknown [103.251.226.246]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id 4CDE51924; Thu, 26 Aug 2021 23:30:29 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1630013430; bh=zSSi/qzqcSSLZaa7u1Ebf/V66KKjje2PliCj7Jnvjdo=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=H0pr7PhjomyUqRj4e8ay6wVBLkhMEbI+dkl+Lhk/gCtdF8DPDeZDZhrOCB9wv+rui g/1c5J6VHOg/Yr5cigES599dfk5VS/AFfppXCBVwHZcUPdSoCBLSHsXLcslv838UlS kq2QilNfPYRscfHVaccYPppM0Bx4x10GJOZkxWmM= From: Umang Jain To: libcamera-devel@lists.libcamera.org Date: Fri, 27 Aug 2021 03:00:13 +0530 Message-Id: <20210826213016.1829504-3-umang.jain@ideasonboard.com> X-Mailer: git-send-email 2.31.1 In-Reply-To: <20210826213016.1829504-1-umang.jain@ideasonboard.com> References: <20210826213016.1829504-1-umang.jain@ideasonboard.com> MIME-Version: 1.0 Subject: [libcamera-devel] [RFC PATCH 2/5] android: camera_stream: Pass FrameBuffer pointer instead of reference X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" Pass the libcamera::FrameBuffer pointer to the post-processor instead of passing it by reference. Passing by reference is fine as long as the post processing is done synchronously. However in subsequent commits, the post processing is planned to be moved to a separate thread. This will conflict as the reference argument (in current case 'source') is copied when we will try to invoke a method on the post-processor using Object::invokeMethod(). The cause of conflict is the LIBCAMERA_DISABLE_COPY_AND_MOVE rule applied on the FrameBuffer class. To resolve the conflict, pass in the FrameBuffer pointer instead of reference. This requires changes to the existing PostProcessor interface and all its implemented classes. Signed-off-by: Umang Jain --- src/android/camera_device.cpp | 2 +- src/android/camera_stream.cpp | 2 +- src/android/camera_stream.h | 2 +- src/android/jpeg/encoder.h | 2 +- src/android/jpeg/encoder_libjpeg.cpp | 4 ++-- src/android/jpeg/encoder_libjpeg.h | 2 +- src/android/jpeg/post_processor_jpeg.cpp | 4 ++-- src/android/jpeg/post_processor_jpeg.h | 4 ++-- src/android/jpeg/thumbnailer.cpp | 4 ++-- src/android/jpeg/thumbnailer.h | 2 +- src/android/post_processor.h | 2 +- src/android/yuv/post_processor_yuv.cpp | 18 +++++++++--------- src/android/yuv/post_processor_yuv.h | 4 ++-- 13 files changed, 26 insertions(+), 26 deletions(-) diff --git a/src/android/camera_device.cpp b/src/android/camera_device.cpp index a69b687a..fea59ce6 100644 --- a/src/android/camera_device.cpp +++ b/src/android/camera_device.cpp @@ -1149,7 +1149,7 @@ void CameraDevice::requestComplete(Request *request) continue; } - int ret = cameraStream->process(*src, *buffer.buffer, + int ret = cameraStream->process(src, *buffer.buffer, descriptor.settings_, resultMetadata.get()); /* diff --git a/src/android/camera_stream.cpp b/src/android/camera_stream.cpp index 61b44183..59b1df6e 100644 --- a/src/android/camera_stream.cpp +++ b/src/android/camera_stream.cpp @@ -98,7 +98,7 @@ int CameraStream::configure() return 0; } -int CameraStream::process(const libcamera::FrameBuffer &source, +int CameraStream::process(const libcamera::FrameBuffer *source, buffer_handle_t camera3Dest, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata) diff --git a/src/android/camera_stream.h b/src/android/camera_stream.h index 2dab6c3a..5c232cb6 100644 --- a/src/android/camera_stream.h +++ b/src/android/camera_stream.h @@ -118,7 +118,7 @@ public: libcamera::Stream *stream() const; int configure(); - int process(const libcamera::FrameBuffer &source, + int process(const libcamera::FrameBuffer *source, buffer_handle_t camera3Dest, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata); diff --git a/src/android/jpeg/encoder.h b/src/android/jpeg/encoder.h index a28522f4..7b6140e7 100644 --- a/src/android/jpeg/encoder.h +++ b/src/android/jpeg/encoder.h @@ -18,7 +18,7 @@ public: virtual ~Encoder() = default; virtual int configure(const libcamera::StreamConfiguration &cfg) = 0; - virtual int encode(const libcamera::FrameBuffer &source, + virtual int encode(const libcamera::FrameBuffer *source, libcamera::Span destination, libcamera::Span exifData, unsigned int quality) = 0; diff --git a/src/android/jpeg/encoder_libjpeg.cpp b/src/android/jpeg/encoder_libjpeg.cpp index a7a63601..1c5ec592 100644 --- a/src/android/jpeg/encoder_libjpeg.cpp +++ b/src/android/jpeg/encoder_libjpeg.cpp @@ -178,10 +178,10 @@ void EncoderLibJpeg::compressNV(Span frame) } } -int EncoderLibJpeg::encode(const FrameBuffer &source, Span dest, +int EncoderLibJpeg::encode(const FrameBuffer *source, Span dest, Span exifData, unsigned int quality) { - MappedFrameBuffer frame(&source, MappedFrameBuffer::MapFlag::Read); + MappedFrameBuffer frame(source, MappedFrameBuffer::MapFlag::Read); if (!frame.isValid()) { LOG(JPEG, Error) << "Failed to map FrameBuffer : " << strerror(frame.error()); diff --git a/src/android/jpeg/encoder_libjpeg.h b/src/android/jpeg/encoder_libjpeg.h index 61fbd1a6..92e57b75 100644 --- a/src/android/jpeg/encoder_libjpeg.h +++ b/src/android/jpeg/encoder_libjpeg.h @@ -20,7 +20,7 @@ public: ~EncoderLibJpeg(); int configure(const libcamera::StreamConfiguration &cfg) override; - int encode(const libcamera::FrameBuffer &source, + int encode(const libcamera::FrameBuffer *source, libcamera::Span destination, libcamera::Span exifData, unsigned int quality) override; diff --git a/src/android/jpeg/post_processor_jpeg.cpp b/src/android/jpeg/post_processor_jpeg.cpp index 3160a784..723e8d18 100644 --- a/src/android/jpeg/post_processor_jpeg.cpp +++ b/src/android/jpeg/post_processor_jpeg.cpp @@ -50,7 +50,7 @@ int PostProcessorJpeg::configure(const StreamConfiguration &inCfg, return encoder_->configure(inCfg); } -void PostProcessorJpeg::generateThumbnail(const FrameBuffer &source, +void PostProcessorJpeg::generateThumbnail(const FrameBuffer *source, const Size &targetSize, unsigned int quality, std::vector *thumbnail) @@ -82,7 +82,7 @@ void PostProcessorJpeg::generateThumbnail(const FrameBuffer &source, } } -int PostProcessorJpeg::process(const FrameBuffer &source, +int PostProcessorJpeg::process(const FrameBuffer *source, CameraBuffer *destination, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata) diff --git a/src/android/jpeg/post_processor_jpeg.h b/src/android/jpeg/post_processor_jpeg.h index 6fd31022..c4b2e9ef 100644 --- a/src/android/jpeg/post_processor_jpeg.h +++ b/src/android/jpeg/post_processor_jpeg.h @@ -22,13 +22,13 @@ public: int configure(const libcamera::StreamConfiguration &incfg, const libcamera::StreamConfiguration &outcfg) override; - int process(const libcamera::FrameBuffer &source, + int process(const libcamera::FrameBuffer *source, CameraBuffer *destination, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata) override; private: - void generateThumbnail(const libcamera::FrameBuffer &source, + void generateThumbnail(const libcamera::FrameBuffer *source, const libcamera::Size &targetSize, unsigned int quality, std::vector *thumbnail); diff --git a/src/android/jpeg/thumbnailer.cpp b/src/android/jpeg/thumbnailer.cpp index 79d83926..46575e4e 100644 --- a/src/android/jpeg/thumbnailer.cpp +++ b/src/android/jpeg/thumbnailer.cpp @@ -37,11 +37,11 @@ void Thumbnailer::configure(const Size &sourceSize, PixelFormat pixelFormat) valid_ = true; } -void Thumbnailer::createThumbnail(const FrameBuffer &source, +void Thumbnailer::createThumbnail(const FrameBuffer *source, const Size &targetSize, std::vector *destination) { - MappedFrameBuffer frame(&source, MappedFrameBuffer::MapFlag::Read); + MappedFrameBuffer frame(source, MappedFrameBuffer::MapFlag::Read); if (!frame.isValid()) { LOG(Thumbnailer, Error) << "Failed to map FrameBuffer : " diff --git a/src/android/jpeg/thumbnailer.h b/src/android/jpeg/thumbnailer.h index 4d086c49..0f3caf40 100644 --- a/src/android/jpeg/thumbnailer.h +++ b/src/android/jpeg/thumbnailer.h @@ -19,7 +19,7 @@ public: void configure(const libcamera::Size &sourceSize, libcamera::PixelFormat pixelFormat); - void createThumbnail(const libcamera::FrameBuffer &source, + void createThumbnail(const libcamera::FrameBuffer *source, const libcamera::Size &targetSize, std::vector *dest); const libcamera::PixelFormat &pixelFormat() const { return pixelFormat_; } diff --git a/src/android/post_processor.h b/src/android/post_processor.h index b2b5bd55..5b8f1ab8 100644 --- a/src/android/post_processor.h +++ b/src/android/post_processor.h @@ -23,7 +23,7 @@ public: virtual int configure(const libcamera::StreamConfiguration &inCfg, const libcamera::StreamConfiguration &outCfg) = 0; - virtual int process(const libcamera::FrameBuffer &source, + virtual int process(const libcamera::FrameBuffer *source, CameraBuffer *destination, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata) = 0; diff --git a/src/android/yuv/post_processor_yuv.cpp b/src/android/yuv/post_processor_yuv.cpp index 3e793a58..4b507d33 100644 --- a/src/android/yuv/post_processor_yuv.cpp +++ b/src/android/yuv/post_processor_yuv.cpp @@ -49,7 +49,7 @@ int PostProcessorYuv::configure(const StreamConfiguration &inCfg, return 0; } -int PostProcessorYuv::process(const FrameBuffer &source, +int PostProcessorYuv::process(const FrameBuffer *source, CameraBuffer *destination, [[maybe_unused]] const CameraMetadata &requestMetadata, [[maybe_unused]] CameraMetadata *metadata) @@ -57,7 +57,7 @@ int PostProcessorYuv::process(const FrameBuffer &source, if (!isValidBuffers(source, *destination)) return -EINVAL; - const MappedFrameBuffer sourceMapped(&source, MappedFrameBuffer::MapFlag::Read); + const MappedFrameBuffer sourceMapped(source, MappedFrameBuffer::MapFlag::Read); if (!sourceMapped.isValid()) { LOG(YUV, Error) << "Failed to mmap camera frame buffer"; return -EINVAL; @@ -83,12 +83,12 @@ int PostProcessorYuv::process(const FrameBuffer &source, return 0; } -bool PostProcessorYuv::isValidBuffers(const FrameBuffer &source, +bool PostProcessorYuv::isValidBuffers(const FrameBuffer *source, const CameraBuffer &destination) const { - if (source.planes().size() != 2) { + if (source->planes().size() != 2) { LOG(YUV, Error) << "Invalid number of source planes: " - << source.planes().size(); + << source->planes().size(); return false; } if (destination.numPlanes() != 2) { @@ -97,12 +97,12 @@ bool PostProcessorYuv::isValidBuffers(const FrameBuffer &source, return false; } - if (source.planes()[0].length < sourceLength_[0] || - source.planes()[1].length < sourceLength_[1]) { + if (source->planes()[0].length < sourceLength_[0] || + source->planes()[1].length < sourceLength_[1]) { LOG(YUV, Error) << "The source planes lengths are too small, actual size: {" - << source.planes()[0].length << ", " - << source.planes()[1].length + << source->planes()[0].length << ", " + << source->planes()[1].length << "}, expected size: {" << sourceLength_[0] << ", " << sourceLength_[1] << "}"; diff --git a/src/android/yuv/post_processor_yuv.h b/src/android/yuv/post_processor_yuv.h index f8b1ba23..44a04113 100644 --- a/src/android/yuv/post_processor_yuv.h +++ b/src/android/yuv/post_processor_yuv.h @@ -20,13 +20,13 @@ public: int configure(const libcamera::StreamConfiguration &incfg, const libcamera::StreamConfiguration &outcfg) override; - int process(const libcamera::FrameBuffer &source, + int process(const libcamera::FrameBuffer *source, CameraBuffer *destination, const CameraMetadata &requestMetadata, CameraMetadata *metadata) override; private: - bool isValidBuffers(const libcamera::FrameBuffer &source, + bool isValidBuffers(const libcamera::FrameBuffer *source, const CameraBuffer &destination) const; void calculateLengths(const libcamera::StreamConfiguration &inCfg, const libcamera::StreamConfiguration &outCfg); From patchwork Thu Aug 26 21:30:14 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Umang Jain X-Patchwork-Id: 13518 X-Patchwork-Delegate: umang.jain@ideasonboard.com Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id 2F668BD87D for ; Thu, 26 Aug 2021 21:30:36 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id D8A2668943; Thu, 26 Aug 2021 23:30:35 +0200 (CEST) Authentication-Results: lancelot.ideasonboard.com; dkim=fail reason="signature verification failed" (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="ulKAsixm"; dkim-atps=neutral Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [213.167.242.64]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id EB04968931 for ; Thu, 26 Aug 2021 23:30:31 +0200 (CEST) Received: from perceval.ideasonboard.com (unknown [103.251.226.246]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id 11B42191F; Thu, 26 Aug 2021 23:30:30 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1630013431; bh=M42B9kLwVCcmVbGJtLD4xOGosrGKST6FW4kQbHpYZZE=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=ulKAsixml9dC1bF+mFTQNH9Y+cDMvkIzkJuFdxPLm7HfLlnWuYQY0YOidRJ3tqBv9 FyGI1HM8IPc5kU1wjmu51v5MuEWnWRC3L4Rg1MLKzOZ4HyEqgSN3pg1YOQqBr23RR6 rGoHf4hJiQYm9aVEe8+RvTepltGL6OxmzAEqAmCU= From: Umang Jain To: libcamera-devel@lists.libcamera.org Date: Fri, 27 Aug 2021 03:00:14 +0530 Message-Id: <20210826213016.1829504-4-umang.jain@ideasonboard.com> X-Mailer: git-send-email 2.31.1 In-Reply-To: <20210826213016.1829504-1-umang.jain@ideasonboard.com> References: <20210826213016.1829504-1-umang.jain@ideasonboard.com> MIME-Version: 1.0 Subject: [libcamera-devel] [RFC PATCH 3/5] android: post_processor: Notify post processing completion status X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" When a post-processor is running asynchronously, we should be able to know whether the post-processing operation on the stream has been completed or not. Secondly, we should be able to know if it completed successfully or encountered some errors. This commit introduces a Signal<> which when connected, will notify that the post-processing has been completed. The status of PostProcessor::process() will be passed as a PostProcessor::Status argument. The populated CameraMetadata shall also be available in the signal-handler. /* * DNI: I do not like the multiple places of signal emission. * We need to use goto but first we need to make a single point * of return somehow, and restructure the error paths a bit. */ Signed-off-by: Umang Jain --- src/android/jpeg/post_processor_jpeg.cpp | 9 ++++++++- src/android/post_processor.h | 7 +++++++ src/android/yuv/post_processor_yuv.cpp | 4 ++++ 3 files changed, 19 insertions(+), 1 deletion(-) diff --git a/src/android/jpeg/post_processor_jpeg.cpp b/src/android/jpeg/post_processor_jpeg.cpp index 723e8d18..9366934d 100644 --- a/src/android/jpeg/post_processor_jpeg.cpp +++ b/src/android/jpeg/post_processor_jpeg.cpp @@ -87,8 +87,11 @@ int PostProcessorJpeg::process(const FrameBuffer *source, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata) { - if (!encoder_) + if (!encoder_) { + processComplete.emit(PostProcessor::Status::Failed, + resultMetadata); return 0; + } ASSERT(destination->numPlanes() == 1); @@ -181,6 +184,8 @@ int PostProcessorJpeg::process(const FrameBuffer *source, exif.data(), quality); if (jpeg_size < 0) { LOG(JPEG, Error) << "Failed to encode stream image"; + processComplete.emit(PostProcessor::Status::Failed, + resultMetadata); return jpeg_size; } @@ -195,5 +200,7 @@ int PostProcessorJpeg::process(const FrameBuffer *source, /* Update the JPEG result Metadata. */ resultMetadata->addEntry(ANDROID_JPEG_SIZE, jpeg_size); + processComplete.emit(PostProcessor::Status::Success, resultMetadata); + return 0; } diff --git a/src/android/post_processor.h b/src/android/post_processor.h index 5b8f1ab8..73fa79e9 100644 --- a/src/android/post_processor.h +++ b/src/android/post_processor.h @@ -8,6 +8,7 @@ #define __ANDROID_POST_PROCESSOR_H__ #include +#include #include #include @@ -27,6 +28,12 @@ public: CameraBuffer *destination, const CameraMetadata &requestMetadata, CameraMetadata *resultMetadata) = 0; + + enum Status { + Success, + Failed, + }; + libcamera::Signal processComplete; }; #endif /* __ANDROID_POST_PROCESSOR_H__ */ diff --git a/src/android/yuv/post_processor_yuv.cpp b/src/android/yuv/post_processor_yuv.cpp index 4b507d33..1877e54b 100644 --- a/src/android/yuv/post_processor_yuv.cpp +++ b/src/android/yuv/post_processor_yuv.cpp @@ -60,6 +60,7 @@ int PostProcessorYuv::process(const FrameBuffer *source, const MappedFrameBuffer sourceMapped(source, MappedFrameBuffer::MapFlag::Read); if (!sourceMapped.isValid()) { LOG(YUV, Error) << "Failed to mmap camera frame buffer"; + processComplete.emit(PostProcessor::Status::Failed, metadata); return -EINVAL; } @@ -77,9 +78,12 @@ int PostProcessorYuv::process(const FrameBuffer *source, libyuv::FilterMode::kFilterBilinear); if (ret) { LOG(YUV, Error) << "Failed NV12 scaling: " << ret; + processComplete.emit(PostProcessor::Status::Failed, metadata); return -EINVAL; } + processComplete.emit(PostProcessor::Status::Success, metadata); + return 0; } From patchwork Thu Aug 26 21:30:15 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Umang Jain X-Patchwork-Id: 13519 X-Patchwork-Delegate: umang.jain@ideasonboard.com Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id 8AABEC3241 for ; Thu, 26 Aug 2021 21:30:36 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id 3943A68946; Thu, 26 Aug 2021 23:30:36 +0200 (CEST) Authentication-Results: lancelot.ideasonboard.com; dkim=fail reason="signature verification failed" (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="RMTxw7wu"; dkim-atps=neutral Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [IPv6:2001:4b98:dc2:55:216:3eff:fef7:d647]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id EC0B268939 for ; Thu, 26 Aug 2021 23:30:33 +0200 (CEST) Received: from perceval.ideasonboard.com (unknown [103.251.226.246]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id E28161924; Thu, 26 Aug 2021 23:30:32 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1630013433; bh=44YKH3zVrd0HTQ8zZnbsoGzcW6FpaIMi3DHZizbS9A8=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=RMTxw7wu+Bt7Ls4AESXCr58IP2IbxkPJZQKPA4hFAPwwGJhwmcz7sWM/0RSt3dwoj 4XKxEUXwLv7HDzYhVfy/g4PiqDrCjq+dN8/OaeWNLERCSfFwR1ooNRpBN3C8bUnueq ITEasP3X1idyqWTE4qD8hh3Q4tO0gfL2vSfuJCL0= From: Umang Jain To: libcamera-devel@lists.libcamera.org Date: Fri, 27 Aug 2021 03:00:15 +0530 Message-Id: <20210826213016.1829504-5-umang.jain@ideasonboard.com> X-Mailer: git-send-email 2.31.1 In-Reply-To: <20210826213016.1829504-1-umang.jain@ideasonboard.com> References: <20210826213016.1829504-1-umang.jain@ideasonboard.com> MIME-Version: 1.0 Subject: [libcamera-devel] [RFC PATCH 4/5] android: camera_stream: Notify process() status with a Signal X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" CameraStream takes care of any post-processing required via the CameraStream::process(). Since the post-processor will be moved to a separate thread (in subsequent commit), the caller of CameraStream::process() should be able to get notified about the status, in order to proceed further for completing the callbacks to android framework. Therefore, chain up to PostProcessor::processComplete signal in its signal handler. Currently, we have only one post-processor in-use, hence emit the corresponding CameraStream::PostProcessing value according to the PostProcessor::status received in the signal handler. Signed-off-by: Umang Jain --- src/android/camera_stream.cpp | 17 +++++++++++++++++ src/android/camera_stream.h | 14 +++++++++++++- 2 files changed, 30 insertions(+), 1 deletion(-) diff --git a/src/android/camera_stream.cpp b/src/android/camera_stream.cpp index 59b1df6e..bdcc7cf9 100644 --- a/src/android/camera_stream.cpp +++ b/src/android/camera_stream.cpp @@ -81,6 +81,8 @@ int CameraStream::configure() int ret = postProcessor_->configure(configuration(), output); if (ret) return ret; + + postProcessor_->processComplete.connect(this, &CameraStream::handlePostProcessing); } if (allocator_) { @@ -119,6 +121,21 @@ int CameraStream::process(const libcamera::FrameBuffer *source, return postProcessor_->process(source, &dest, requestMetadata, resultMetadata); } +void CameraStream::handlePostProcessing(PostProcessor::Status status, + CameraMetadata *resultMetadata) +{ + switch (status) { + case PostProcessor::Status::Success: + processComplete.emit(this, ProcessStatus::Success, resultMetadata); + break; + case PostProcessor::Status::Failed: + processComplete.emit(this, ProcessStatus::Failed, resultMetadata); + break; + default: + LOG(HAL, Error) << "PostProcessor status invalid"; + } +} + FrameBuffer *CameraStream::getBuffer() { if (!allocator_) diff --git a/src/android/camera_stream.h b/src/android/camera_stream.h index 5c232cb6..d54d3f58 100644 --- a/src/android/camera_stream.h +++ b/src/android/camera_stream.h @@ -13,15 +13,18 @@ #include +#include + #include #include #include #include #include +#include "post_processor.h" + class CameraDevice; class CameraMetadata; -class PostProcessor; class CameraStream { @@ -125,7 +128,16 @@ public: libcamera::FrameBuffer *getBuffer(); void putBuffer(libcamera::FrameBuffer *buffer); + enum ProcessStatus { + Failed, + Success, + }; + libcamera::Signal processComplete; + private: + void handlePostProcessing(PostProcessor::Status status, + CameraMetadata *resultMetadata); + CameraDevice *const cameraDevice_; const libcamera::CameraConfiguration *config_; const Type type_; From patchwork Thu Aug 26 21:30:16 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Umang Jain X-Patchwork-Id: 13520 X-Patchwork-Delegate: umang.jain@ideasonboard.com Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id A4AFEC3242 for ; Thu, 26 Aug 2021 21:30:39 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id 1E3DA6893B; Thu, 26 Aug 2021 23:30:39 +0200 (CEST) Authentication-Results: lancelot.ideasonboard.com; dkim=fail reason="signature verification failed" (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="kWGv9kJ9"; dkim-atps=neutral Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [IPv6:2001:4b98:dc2:55:216:3eff:fef7:d647]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id 18E8068945 for ; Thu, 26 Aug 2021 23:30:36 +0200 (CEST) Received: from perceval.ideasonboard.com (unknown [103.251.226.246]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id EA780191F; Thu, 26 Aug 2021 23:30:34 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1630013435; bh=4269RlCIKL1mKXTIIBzoep+cFdGCu4eEmr/5A5CHn0Q=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=kWGv9kJ9wbbe+PzjDOOmqz0g4E0wbQ3SANhdDXk/zc2BZG7wmkZ4mtJOX8NY7QIjI IzlM5Z4EVD86J2rYiCfZAawZWoG1YcBCbqLJujwUEmdua+A+k+YS+JP7Mtn/W+4oWy o9j3lONKUuTF0+Q7kepkEEb5/RQUM6O3S1DK+dmQ= From: Umang Jain To: libcamera-devel@lists.libcamera.org Date: Fri, 27 Aug 2021 03:00:16 +0530 Message-Id: <20210826213016.1829504-6-umang.jain@ideasonboard.com> X-Mailer: git-send-email 2.31.1 In-Reply-To: <20210826213016.1829504-1-umang.jain@ideasonboard.com> References: <20210826213016.1829504-1-umang.jain@ideasonboard.com> MIME-Version: 1.0 Subject: [libcamera-devel] [RFC PATCH 5/5] android: Run post processing in a separate thread X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" In CameraStream, introduce a new private PostProcessorWorker class derived from Thread. The post-processor shall run in this derived thread instance asynchronously. Running PostProcessor asynchronously should entail that all the data needed by the PostProcessor should remain valid for the entire duration of its run. In this instance, we currently need to ensure: - 'source' FrameBuffer for the post processor - Camera result metadata Should be valid and saved with preserving the entire context. The 'source' framebuffer is copied internally for streams other than internal (since internal ones are allocated by CameraStream class itself) and the copy is passed along to post-processor. Coming to preserving the context, a quick reference on how captured results are sent to android framework. Completed captured results should be sent using process_capture_result() callback. The order of sending them should match the order the capture request (camera3_capture_request_t), that was submitted to the HAL in the first place. In order to enforce the ordering, we need to maintain a queue. When a stream requires post-processing, we save the associated capture results (via Camera3RequestDescriptor) and add it to the queue. All transient completed captured results are queued as well. When the post-processing results are available, we can simply start de-queuing all the queued completed captured results and invoke process_capture_result() callback on each of them. The context saving part is done by Camera3RequestDescriptor. It is further extended to include data-structs to fulfill the demands of async post-processing. Signed-off-by: Umang Jain --- src/android/camera_device.cpp | 92 +++++++++++++++++++++++++++++++---- src/android/camera_device.h | 21 +++++++- src/android/camera_stream.cpp | 35 ++++++++++--- src/android/camera_stream.h | 40 +++++++++++++++ 4 files changed, 170 insertions(+), 18 deletions(-) diff --git a/src/android/camera_device.cpp b/src/android/camera_device.cpp index fea59ce6..08237187 100644 --- a/src/android/camera_device.cpp +++ b/src/android/camera_device.cpp @@ -960,6 +960,7 @@ int CameraDevice::processCaptureRequest(camera3_capture_request_t *camera3Reques * and add them to the Request if required. */ FrameBuffer *buffer = nullptr; + descriptor.srcInternalBuffer_ = nullptr; switch (cameraStream->type()) { case CameraStream::Type::Mapped: /* @@ -990,6 +991,7 @@ int CameraDevice::processCaptureRequest(camera3_capture_request_t *camera3Reques * once it has been processed. */ buffer = cameraStream->getBuffer(); + descriptor.srcInternalBuffer_ = buffer; LOG(HAL, Debug) << ss.str() << " (internal)"; break; } @@ -1149,25 +1151,95 @@ void CameraDevice::requestComplete(Request *request) continue; } + /* + * Save the current context of capture result and queue the + * descriptor. cameraStream will process asynchronously, so + * we can only send the results back after the processing has + * been completed. + */ + descriptor.status_ = Camera3RequestDescriptor::NOT_READY; + descriptor.resultMetadata_ = std::move(resultMetadata); + descriptor.captureResult_ = captureResult; + + cameraStream->processComplete.connect(this, &CameraDevice::streamProcessingComplete); int ret = cameraStream->process(src, *buffer.buffer, descriptor.settings_, - resultMetadata.get()); + descriptor.resultMetadata_.get()); + std::unique_ptr reqDescriptor = + std::make_unique(); + *reqDescriptor = std::move(descriptor); + queuedDescriptor_.push_back(std::move(reqDescriptor)); + + return; + } + + if (queuedDescriptor_.empty()) { + captureResult.result = resultMetadata->get(); + callbacks_->process_capture_result(callbacks_, &captureResult); + } else { + /* + * Previous capture results waiting to be sent to framework + * hence, queue the current capture results as well. After that, + * check if any results are ready to be sent back to the + * framework. + */ + descriptor.status_ = Camera3RequestDescriptor::READY_SUCCESS; + descriptor.resultMetadata_ = std::move(resultMetadata); + descriptor.captureResult_ = captureResult; + std::unique_ptr reqDescriptor = + std::make_unique(); + *reqDescriptor = std::move(descriptor); + queuedDescriptor_.push_back(std::move(reqDescriptor)); + + while (!queuedDescriptor_.empty()) { + std::unique_ptr &d = queuedDescriptor_.front(); + if (d->status_ != Camera3RequestDescriptor::NOT_READY) { + d->captureResult_.result = d->resultMetadata_->get(); + callbacks_->process_capture_result(callbacks_, &(d->captureResult_)); + queuedDescriptor_.pop_front(); + } else { + break; + } + } + } +} + +void CameraDevice::streamProcessingComplete(CameraStream *cameraStream, + CameraStream::ProcessStatus status, + CameraMetadata *resultMetadata) +{ + cameraStream->processComplete.disconnect(this, &CameraDevice::streamProcessingComplete); + + for (auto &d : queuedDescriptor_) { + if (d->resultMetadata_.get() != resultMetadata) + continue; + /* * Return the FrameBuffer to the CameraStream now that we're * done processing it. */ if (cameraStream->type() == CameraStream::Type::Internal) - cameraStream->putBuffer(src); - - if (ret) { - buffer.status = CAMERA3_BUFFER_STATUS_ERROR; - notifyError(descriptor.frameNumber_, buffer.stream, - CAMERA3_MSG_ERROR_BUFFER); + cameraStream->putBuffer(d->srcInternalBuffer_); + + if (status == CameraStream::ProcessStatus::Success) { + d->status_ = Camera3RequestDescriptor::READY_SUCCESS; + } else { + /* \todo this is clumsy error handling, be better. */ + d->status_ = Camera3RequestDescriptor::READY_FAILED; + for (camera3_stream_buffer_t &buffer : d->buffers_) { + CameraStream *cs = static_cast(buffer.stream->priv); + + if (cs->camera3Stream().format != HAL_PIXEL_FORMAT_BLOB) + continue; + + buffer.status = CAMERA3_BUFFER_STATUS_ERROR; + notifyError(d->frameNumber_, buffer.stream, + CAMERA3_MSG_ERROR_BUFFER); + } } - } - captureResult.result = resultMetadata->get(); - callbacks_->process_capture_result(callbacks_, &captureResult); + return; + } } std::string CameraDevice::logPrefix() const diff --git a/src/android/camera_device.h b/src/android/camera_device.h index dd9aebba..4e4bb970 100644 --- a/src/android/camera_device.h +++ b/src/android/camera_device.h @@ -7,6 +7,7 @@ #ifndef __ANDROID_CAMERA_DEVICE_H__ #define __ANDROID_CAMERA_DEVICE_H__ +#include #include #include #include @@ -81,6 +82,20 @@ private: std::vector> frameBuffers_; CameraMetadata settings_; std::unique_ptr request_; + + /* + * Only set when a capture result needs to be queued before + * completion via process_capture_result() callback. + */ + enum processingStatus { + NOT_READY, + READY_SUCCESS, + READY_FAILED, + }; + std::unique_ptr resultMetadata_; + camera3_capture_result_t captureResult_; + libcamera::FrameBuffer *srcInternalBuffer_; + processingStatus status_; }; enum class State { @@ -100,7 +115,9 @@ private: int processControls(Camera3RequestDescriptor *descriptor); std::unique_ptr getResultMetadata( const Camera3RequestDescriptor &descriptor) const; - + void streamProcessingComplete(CameraStream *cameraStream, + CameraStream::ProcessStatus status, + CameraMetadata *resultMetadata); unsigned int id_; camera3_device_t camera3Device_; @@ -128,6 +145,8 @@ private: int orientation_; CameraMetadata lastSettings_; + + std::deque> queuedDescriptor_; }; #endif /* __ANDROID_CAMERA_DEVICE_H__ */ diff --git a/src/android/camera_stream.cpp b/src/android/camera_stream.cpp index bdcc7cf9..d5981c0e 100644 --- a/src/android/camera_stream.cpp +++ b/src/android/camera_stream.cpp @@ -55,6 +55,7 @@ CameraStream::CameraStream(CameraDevice *const cameraDevice, * is what we instantiate here. */ postProcessor_ = std::make_unique(cameraDevice_); + ppWorker_ = std::make_unique(postProcessor_.get()); } if (type == Type::Internal) { @@ -108,22 +109,41 @@ int CameraStream::process(const libcamera::FrameBuffer *source, if (!postProcessor_) return 0; - /* - * \todo Buffer mapping and processing should be moved to a - * separate thread. - */ - CameraBuffer dest(camera3Dest, PROT_READ | PROT_WRITE); - if (!dest.isValid()) { + /* \todo Should buffer mapping be moved to post-processor's thread? */ + dest_ = std::make_unique(camera3Dest, PROT_READ | PROT_WRITE); + if (!dest_->isValid()) { LOG(HAL, Error) << "Failed to map android blob buffer"; return -EINVAL; } - return postProcessor_->process(source, &dest, requestMetadata, resultMetadata); + if (type() != CameraStream::Type::Internal) { + /* + * The source buffer is owned by Request object which can be + * reused for future capture request. Since post-processor will + * run asynchrnously, we need to copy the source buffer and use + * it as source data for post-processor. + */ + srcPlanes_.clear(); + for (const auto &plane : source->planes()) + srcPlanes_.push_back(plane); + + srcFramebuffer_ = std::make_unique(srcPlanes_); + source = srcFramebuffer_.get(); + } + + ppWorker_->start(); + + return postProcessor_->invokeMethod(&PostProcessor::process, + ConnectionTypeQueued, + source, dest_.get(), + requestMetadata, resultMetadata); } void CameraStream::handlePostProcessing(PostProcessor::Status status, CameraMetadata *resultMetadata) { + ppWorker_->exit(); + switch (status) { case PostProcessor::Status::Success: processComplete.emit(this, ProcessStatus::Success, resultMetadata); @@ -134,6 +154,7 @@ void CameraStream::handlePostProcessing(PostProcessor::Status status, default: LOG(HAL, Error) << "PostProcessor status invalid"; } + srcFramebuffer_.reset(); } FrameBuffer *CameraStream::getBuffer() diff --git a/src/android/camera_stream.h b/src/android/camera_stream.h index d54d3f58..567d896f 100644 --- a/src/android/camera_stream.h +++ b/src/android/camera_stream.h @@ -13,7 +13,9 @@ #include +#include #include +#include #include #include @@ -23,6 +25,7 @@ #include "post_processor.h" +class CameraBuffer; class CameraDevice; class CameraMetadata; @@ -135,6 +138,38 @@ public: libcamera::Signal processComplete; private: + class PostProcessorWorker : public libcamera::Thread + { + public: + PostProcessorWorker(PostProcessor *postProcessor) + { + postProcessor->moveToThread(this); + } + + void start() + { + /* + * \todo [BLOCKER] !!! + * On second shutter capture, this fails to start the + * thread(again). No errors for first shutter capture. + */ + Thread::start(); + } + + void stop() + { + exit(); + wait(); + } + + protected: + void run() override + { + exec(); + dispatchMessages(); + } + }; + void handlePostProcessing(PostProcessor::Status status, CameraMetadata *resultMetadata); @@ -152,6 +187,11 @@ private: */ std::unique_ptr mutex_; std::unique_ptr postProcessor_; + std::unique_ptr ppWorker_; + + std::unique_ptr dest_; + std::unique_ptr srcFramebuffer_; + std::vector srcPlanes_; }; #endif /* __ANDROID_CAMERA_STREAM__ */