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