[libcamera-devel,v3,7/8] libcamera: pipeline: rkisp1: Use SOF event to warn about late parameters
diff mbox series

Message ID 20201123221234.485933-8-niklas.soderlund@ragnatech.se
State Superseded
Delegated to: Niklas Söderlund
Headers show
Series
  • libcamera: Add helper for controls that take effect with a delay
Related show

Commit Message

Niklas Söderlund Nov. 23, 2020, 10:12 p.m. UTC
In the Timeline approach the idea was to delay queuing buffers to the
device until the IPA had a chance to prepare the parameters buffer. A
check was still added to warn if the IPA queued buffers before the
parameters buffer was filled in.

This check happened much sooner then needed as the parameter buffers
does not have to be ready when the buffer is queued but just before its
consumed. As the pipeline now has true knowledge of each SOF we can move
the check there and remove the delaying of queuing of buffers.

This change really speeds up the IPA reactions as the delays used in the
Timeline where approximated while with this change they are driven by
events reported by the device.

Signed-off-by: Niklas Söderlund <niklas.soderlund@ragnatech.se>
---
 src/libcamera/pipeline/rkisp1/rkisp1.cpp | 78 ++++++++----------------
 1 file changed, 24 insertions(+), 54 deletions(-)

Comments

Jacopo Mondi Nov. 27, 2020, 11:05 a.m. UTC | #1
Hi Niklas,

On Mon, Nov 23, 2020 at 11:12:33PM +0100, Niklas Söderlund wrote:
> In the Timeline approach the idea was to delay queuing buffers to the
> device until the IPA had a chance to prepare the parameters buffer. A
> check was still added to warn if the IPA queued buffers before the
> parameters buffer was filled in.
>
> This check happened much sooner then needed as the parameter buffers
> does not have to be ready when the buffer is queued but just before its
> consumed. As the pipeline now has true knowledge of each SOF we can move
> the check there and remove the delaying of queuing of buffers.
>
> This change really speeds up the IPA reactions as the delays used in the
> Timeline where approximated while with this change they are driven by
> events reported by the device.
>
> Signed-off-by: Niklas Söderlund <niklas.soderlund@ragnatech.se>
> ---
>  src/libcamera/pipeline/rkisp1/rkisp1.cpp | 78 ++++++++----------------
>  1 file changed, 24 insertions(+), 54 deletions(-)
>
> diff --git a/src/libcamera/pipeline/rkisp1/rkisp1.cpp b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> index c3c4b5a65e3d9afe..3662a53ac4a43fcd 100644
> --- a/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> +++ b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> @@ -40,7 +40,6 @@ namespace libcamera {
>  LOG_DEFINE_CATEGORY(RkISP1)
>
>  class PipelineHandlerRkISP1;
> -class RkISP1ActionQueueBuffers;
>  class RkISP1CameraData;
>
>  enum RkISP1ActionType {
> @@ -203,7 +202,6 @@ private:
>  			PipelineHandler::cameraData(camera));
>  	}
>
> -	friend RkISP1ActionQueueBuffers;
>  	friend RkISP1CameraData;
>  	friend RkISP1Frames;
>
> @@ -214,6 +212,7 @@ private:
>  	void bufferReady(FrameBuffer *buffer);
>  	void paramReady(FrameBuffer *buffer);
>  	void statReady(FrameBuffer *buffer);
> +	void frameStart(uint32_t sequence);
>
>  	int allocateBuffers(Camera *camera);
>  	int freeBuffers(Camera *camera);
> @@ -348,53 +347,6 @@ RkISP1FrameInfo *RkISP1Frames::find(Request *request)
>  	return nullptr;
>  }
>
> -class RkISP1ActionQueueBuffers : public FrameAction
> -{
> -public:
> -	RkISP1ActionQueueBuffers(unsigned int frame, RkISP1CameraData *data,
> -				 PipelineHandlerRkISP1 *pipe)
> -		: FrameAction(frame, QueueBuffers), data_(data), pipe_(pipe)
> -	{
> -	}
> -
> -protected:
> -	void run() override
> -	{
> -		RkISP1FrameInfo *info = data_->frameInfo_.find(frame());
> -		if (!info)
> -			LOG(RkISP1, Fatal) << "Frame not known";
> -
> -		/*
> -		 * \todo: If parameters are not filled a better method to handle
> -		 * the situation than queuing a buffer with unknown content
> -		 * should be used.
> -		 *
> -		 * It seems excessive to keep an internal zeroed scratch
> -		 * parameters buffer around as this should not happen unless the
> -		 * devices is under too much load. Perhaps failing the request
> -		 * and returning it to the application with an error code is
> -		 * better than queue it to hardware?
> -		 */
> -		if (!info->paramFilled)
> -			LOG(RkISP1, Error)
> -				<< "Parameters not ready on time for frame "
> -				<< frame();
> -
> -		pipe_->param_->queueBuffer(info->paramBuffer);
> -		pipe_->stat_->queueBuffer(info->statBuffer);
> -
> -		if (info->mainPathBuffer)
> -			pipe_->mainPath_.queueBuffer(info->mainPathBuffer);
> -
> -		if (info->selfPathBuffer)
> -			pipe_->selfPath_.queueBuffer(info->selfPathBuffer);
> -	}
> -
> -private:
> -	RkISP1CameraData *data_;
> -	PipelineHandlerRkISP1 *pipe_;
> -};
> -
>  int RkISP1CameraData::loadIPA()
>  {
>  	ipa_ = IPAManager::createIPA(pipe_, 1, 1);
> @@ -958,9 +910,14 @@ int PipelineHandlerRkISP1::queueRequestDevice(Camera *camera, Request *request)
>  	op.controls = { request->controls() };
>  	data->ipa_->processEvent(op);
>
> -	data->timeline_.scheduleAction(std::make_unique<RkISP1ActionQueueBuffers>(data->frame_,
> -										  data,
> -										  this));
> +	param_->queueBuffer(info->paramBuffer);
> +	stat_->queueBuffer(info->statBuffer);
> +
> +	if (info->mainPathBuffer)
> +		mainPath_.queueBuffer(info->mainPathBuffer);
> +
> +	if (info->selfPathBuffer)
> +		selfPath_.queueBuffer(info->selfPathBuffer);
>
>  	data->frame_++;
>
> @@ -1098,6 +1055,7 @@ bool PipelineHandlerRkISP1::match(DeviceEnumerator *enumerator)
>  	mainPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
>  	selfPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
>  	stat_->bufferReady.connect(this, &PipelineHandlerRkISP1::statReady);
> +	isp_->frameStart.connect(this, &PipelineHandlerRkISP1::frameStart);
>  	param_->bufferReady.connect(this, &PipelineHandlerRkISP1::paramReady);
>
>  	/*
> @@ -1177,8 +1135,6 @@ void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
>  	if (!info)
>  		return;
>
> -	data->timeline_.bufferReady(buffer);
> -
>  	if (data->frame_ <= buffer->metadata().sequence)
>  		data->frame_ = buffer->metadata().sequence + 1;
>
> @@ -1188,6 +1144,20 @@ void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
>  	data->ipa_->processEvent(op);
>  }
>
> +void PipelineHandlerRkISP1::frameStart(uint32_t sequence)
> +{
> +	if (!activeCamera_)
> +		return;
> +
> +	RkISP1CameraData *data = cameraData(activeCamera_);
> +
> +	RkISP1FrameInfo *info = data->frameInfo_.find(sequence);
> +	if (!info || !info->paramFilled)
> +		LOG(RkISP1, Info)
> +			<< "Parameters not ready on time for frame "
> +			<< sequence;
> +}
> +

This surely is my lack of understanding, but shouldn't we worry about
this a param buffer queue time ?

>  REGISTER_PIPELINE_HANDLER(PipelineHandlerRkISP1)
>
>  } /* namespace libcamera */
> --
> 2.29.2
>
> _______________________________________________
> libcamera-devel mailing list
> libcamera-devel@lists.libcamera.org
> https://lists.libcamera.org/listinfo/libcamera-devel
Laurent Pinchart Dec. 5, 2020, 2:08 a.m. UTC | #2
Hi Niklas,

Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>

On Fri, Nov 27, 2020 at 12:05:10PM +0100, Jacopo Mondi wrote:
> On Mon, Nov 23, 2020 at 11:12:33PM +0100, Niklas Söderlund wrote:
> > In the Timeline approach the idea was to delay queuing buffers to the
> > device until the IPA had a chance to prepare the parameters buffer. A
> > check was still added to warn if the IPA queued buffers before the
> > parameters buffer was filled in.
> >
> > This check happened much sooner then needed as the parameter buffers
> > does not have to be ready when the buffer is queued but just before its

s/its/it's/

> > consumed. As the pipeline now has true knowledge of each SOF we can move
> > the check there and remove the delaying of queuing of buffers.
> >
> > This change really speeds up the IPA reactions as the delays used in the
> > Timeline where approximated while with this change they are driven by
> > events reported by the device.
> >
> > Signed-off-by: Niklas Söderlund <niklas.soderlund@ragnatech.se>
> > ---
> >  src/libcamera/pipeline/rkisp1/rkisp1.cpp | 78 ++++++++----------------
> >  1 file changed, 24 insertions(+), 54 deletions(-)
> >
> > diff --git a/src/libcamera/pipeline/rkisp1/rkisp1.cpp b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> > index c3c4b5a65e3d9afe..3662a53ac4a43fcd 100644
> > --- a/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> > +++ b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> > @@ -40,7 +40,6 @@ namespace libcamera {
> >  LOG_DEFINE_CATEGORY(RkISP1)
> >
> >  class PipelineHandlerRkISP1;
> > -class RkISP1ActionQueueBuffers;
> >  class RkISP1CameraData;
> >
> >  enum RkISP1ActionType {
> > @@ -203,7 +202,6 @@ private:
> >  			PipelineHandler::cameraData(camera));
> >  	}
> >
> > -	friend RkISP1ActionQueueBuffers;
> >  	friend RkISP1CameraData;
> >  	friend RkISP1Frames;
> >
> > @@ -214,6 +212,7 @@ private:
> >  	void bufferReady(FrameBuffer *buffer);
> >  	void paramReady(FrameBuffer *buffer);
> >  	void statReady(FrameBuffer *buffer);
> > +	void frameStart(uint32_t sequence);
> >
> >  	int allocateBuffers(Camera *camera);
> >  	int freeBuffers(Camera *camera);
> > @@ -348,53 +347,6 @@ RkISP1FrameInfo *RkISP1Frames::find(Request *request)
> >  	return nullptr;
> >  }
> >
> > -class RkISP1ActionQueueBuffers : public FrameAction
> > -{
> > -public:
> > -	RkISP1ActionQueueBuffers(unsigned int frame, RkISP1CameraData *data,
> > -				 PipelineHandlerRkISP1 *pipe)
> > -		: FrameAction(frame, QueueBuffers), data_(data), pipe_(pipe)
> > -	{
> > -	}
> > -
> > -protected:
> > -	void run() override
> > -	{
> > -		RkISP1FrameInfo *info = data_->frameInfo_.find(frame());
> > -		if (!info)
> > -			LOG(RkISP1, Fatal) << "Frame not known";
> > -
> > -		/*
> > -		 * \todo: If parameters are not filled a better method to handle
> > -		 * the situation than queuing a buffer with unknown content
> > -		 * should be used.
> > -		 *
> > -		 * It seems excessive to keep an internal zeroed scratch
> > -		 * parameters buffer around as this should not happen unless the
> > -		 * devices is under too much load. Perhaps failing the request
> > -		 * and returning it to the application with an error code is
> > -		 * better than queue it to hardware?
> > -		 */
> > -		if (!info->paramFilled)
> > -			LOG(RkISP1, Error)
> > -				<< "Parameters not ready on time for frame "
> > -				<< frame();
> > -
> > -		pipe_->param_->queueBuffer(info->paramBuffer);
> > -		pipe_->stat_->queueBuffer(info->statBuffer);
> > -
> > -		if (info->mainPathBuffer)
> > -			pipe_->mainPath_.queueBuffer(info->mainPathBuffer);
> > -
> > -		if (info->selfPathBuffer)
> > -			pipe_->selfPath_.queueBuffer(info->selfPathBuffer);
> > -	}
> > -
> > -private:
> > -	RkISP1CameraData *data_;
> > -	PipelineHandlerRkISP1 *pipe_;
> > -};
> > -
> >  int RkISP1CameraData::loadIPA()
> >  {
> >  	ipa_ = IPAManager::createIPA(pipe_, 1, 1);
> > @@ -958,9 +910,14 @@ int PipelineHandlerRkISP1::queueRequestDevice(Camera *camera, Request *request)
> >  	op.controls = { request->controls() };
> >  	data->ipa_->processEvent(op);
> >
> > -	data->timeline_.scheduleAction(std::make_unique<RkISP1ActionQueueBuffers>(data->frame_,
> > -										  data,
> > -										  this));
> > +	param_->queueBuffer(info->paramBuffer);
> > +	stat_->queueBuffer(info->statBuffer);
> > +
> > +	if (info->mainPathBuffer)
> > +		mainPath_.queueBuffer(info->mainPathBuffer);
> > +
> > +	if (info->selfPathBuffer)
> > +		selfPath_.queueBuffer(info->selfPathBuffer);

You're queuing the buffers to the ISP, including the ISP parameters,
before the IPA gets a change to even take into account the parameters
for this request to compute the corresponding ISP parameters. Do we
really want to do that ?

> >
> >  	data->frame_++;
> >
> > @@ -1098,6 +1055,7 @@ bool PipelineHandlerRkISP1::match(DeviceEnumerator *enumerator)
> >  	mainPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
> >  	selfPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
> >  	stat_->bufferReady.connect(this, &PipelineHandlerRkISP1::statReady);
> > +	isp_->frameStart.connect(this, &PipelineHandlerRkISP1::frameStart);
> >  	param_->bufferReady.connect(this, &PipelineHandlerRkISP1::paramReady);
> >
> >  	/*
> > @@ -1177,8 +1135,6 @@ void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
> >  	if (!info)
> >  		return;
> >
> > -	data->timeline_.bufferReady(buffer);
> > -
> >  	if (data->frame_ <= buffer->metadata().sequence)
> >  		data->frame_ = buffer->metadata().sequence + 1;
> >
> > @@ -1188,6 +1144,20 @@ void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
> >  	data->ipa_->processEvent(op);
> >  }
> >
> > +void PipelineHandlerRkISP1::frameStart(uint32_t sequence)
> > +{
> > +	if (!activeCamera_)
> > +		return;
> > +
> > +	RkISP1CameraData *data = cameraData(activeCamera_);
> > +
> > +	RkISP1FrameInfo *info = data->frameInfo_.find(sequence);
> > +	if (!info || !info->paramFilled)
> > +		LOG(RkISP1, Info)
> > +			<< "Parameters not ready on time for frame "
> > +			<< sequence;
> > +}
> > +
> 
> This surely is my lack of understanding, but shouldn't we worry about
> this a param buffer queue time ?
> 
> >  REGISTER_PIPELINE_HANDLER(PipelineHandlerRkISP1)
> >
> >  } /* namespace libcamera */
Niklas Söderlund Dec. 11, 2020, 4:32 p.m. UTC | #3
Hi Laurent and Jacopo,

Thanks for your feedback. I think you both ask the same thing but in two 
different locations in the file so I reply only at one ;-)

On 2020-12-05 04:08:32 +0200, Laurent Pinchart wrote:
> Hi Niklas,
> 
> Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
> 
> On Fri, Nov 27, 2020 at 12:05:10PM +0100, Jacopo Mondi wrote:
> > On Mon, Nov 23, 2020 at 11:12:33PM +0100, Niklas Söderlund wrote:
> > > In the Timeline approach the idea was to delay queuing buffers to the
> > > device until the IPA had a chance to prepare the parameters buffer. A
> > > check was still added to warn if the IPA queued buffers before the
> > > parameters buffer was filled in.
> > >
> > > This check happened much sooner then needed as the parameter buffers
> > > does not have to be ready when the buffer is queued but just before its
> 
> s/its/it's/
> 
> > > consumed. As the pipeline now has true knowledge of each SOF we can move
> > > the check there and remove the delaying of queuing of buffers.
> > >
> > > This change really speeds up the IPA reactions as the delays used in the
> > > Timeline where approximated while with this change they are driven by
> > > events reported by the device.
> > >
> > > Signed-off-by: Niklas Söderlund <niklas.soderlund@ragnatech.se>
> > > ---
> > >  src/libcamera/pipeline/rkisp1/rkisp1.cpp | 78 ++++++++----------------
> > >  1 file changed, 24 insertions(+), 54 deletions(-)
> > >
> > > diff --git a/src/libcamera/pipeline/rkisp1/rkisp1.cpp b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> > > index c3c4b5a65e3d9afe..3662a53ac4a43fcd 100644
> > > --- a/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> > > +++ b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
> > > @@ -40,7 +40,6 @@ namespace libcamera {
> > >  LOG_DEFINE_CATEGORY(RkISP1)
> > >
> > >  class PipelineHandlerRkISP1;
> > > -class RkISP1ActionQueueBuffers;
> > >  class RkISP1CameraData;
> > >
> > >  enum RkISP1ActionType {
> > > @@ -203,7 +202,6 @@ private:
> > >  			PipelineHandler::cameraData(camera));
> > >  	}
> > >
> > > -	friend RkISP1ActionQueueBuffers;
> > >  	friend RkISP1CameraData;
> > >  	friend RkISP1Frames;
> > >
> > > @@ -214,6 +212,7 @@ private:
> > >  	void bufferReady(FrameBuffer *buffer);
> > >  	void paramReady(FrameBuffer *buffer);
> > >  	void statReady(FrameBuffer *buffer);
> > > +	void frameStart(uint32_t sequence);
> > >
> > >  	int allocateBuffers(Camera *camera);
> > >  	int freeBuffers(Camera *camera);
> > > @@ -348,53 +347,6 @@ RkISP1FrameInfo *RkISP1Frames::find(Request *request)
> > >  	return nullptr;
> > >  }
> > >
> > > -class RkISP1ActionQueueBuffers : public FrameAction
> > > -{
> > > -public:
> > > -	RkISP1ActionQueueBuffers(unsigned int frame, RkISP1CameraData *data,
> > > -				 PipelineHandlerRkISP1 *pipe)
> > > -		: FrameAction(frame, QueueBuffers), data_(data), pipe_(pipe)
> > > -	{
> > > -	}
> > > -
> > > -protected:
> > > -	void run() override
> > > -	{
> > > -		RkISP1FrameInfo *info = data_->frameInfo_.find(frame());
> > > -		if (!info)
> > > -			LOG(RkISP1, Fatal) << "Frame not known";
> > > -
> > > -		/*
> > > -		 * \todo: If parameters are not filled a better method to handle
> > > -		 * the situation than queuing a buffer with unknown content
> > > -		 * should be used.
> > > -		 *
> > > -		 * It seems excessive to keep an internal zeroed scratch
> > > -		 * parameters buffer around as this should not happen unless the
> > > -		 * devices is under too much load. Perhaps failing the request
> > > -		 * and returning it to the application with an error code is
> > > -		 * better than queue it to hardware?
> > > -		 */
> > > -		if (!info->paramFilled)
> > > -			LOG(RkISP1, Error)
> > > -				<< "Parameters not ready on time for frame "
> > > -				<< frame();
> > > -
> > > -		pipe_->param_->queueBuffer(info->paramBuffer);
> > > -		pipe_->stat_->queueBuffer(info->statBuffer);
> > > -
> > > -		if (info->mainPathBuffer)
> > > -			pipe_->mainPath_.queueBuffer(info->mainPathBuffer);
> > > -
> > > -		if (info->selfPathBuffer)
> > > -			pipe_->selfPath_.queueBuffer(info->selfPathBuffer);
> > > -	}
> > > -
> > > -private:
> > > -	RkISP1CameraData *data_;
> > > -	PipelineHandlerRkISP1 *pipe_;
> > > -};
> > > -
> > >  int RkISP1CameraData::loadIPA()
> > >  {
> > >  	ipa_ = IPAManager::createIPA(pipe_, 1, 1);
> > > @@ -958,9 +910,14 @@ int PipelineHandlerRkISP1::queueRequestDevice(Camera *camera, Request *request)
> > >  	op.controls = { request->controls() };
> > >  	data->ipa_->processEvent(op);
> > >
> > > -	data->timeline_.scheduleAction(std::make_unique<RkISP1ActionQueueBuffers>(data->frame_,
> > > -										  data,
> > > -										  this));
> > > +	param_->queueBuffer(info->paramBuffer);
> > > +	stat_->queueBuffer(info->statBuffer);
> > > +
> > > +	if (info->mainPathBuffer)
> > > +		mainPath_.queueBuffer(info->mainPathBuffer);
> > > +
> > > +	if (info->selfPathBuffer)
> > > +		selfPath_.queueBuffer(info->selfPathBuffer);
> 
> You're queuing the buffers to the ISP, including the ISP parameters,
> before the IPA gets a change to even take into account the parameters
> for this request to compute the corresponding ISP parameters. Do we
> really want to do that ?

I'm not sure if this is what we want to do in the end, but as I 
understand it this is how we agreed to do it when this was discussed 
last time and is what the Timeline solution this replaces tried to do.

The rational we had back then was that we need to queue the buffers to 
the video device as soon as possible as to not stall the pipeline. We 
also know the IPA may fill in the buffer up until the point in time 
where it's consumed by the ISP hardware. In this solution we "buy" the 
IPA more time to fill the param buffer then we would if we blocked for 
IPA feedback.

In this pipeline we judge that a param buffer is consume by the hardware 
when we see the corresponding SOE. If we at that time have not had 
feedback form the IPA we print the warning Jacopo comments on below.

I'm not committed to this mode of operation but if we wish to change it 
it think we should do so before or after this series. What do you guys 
think?

> 
> > >
> > >  	data->frame_++;
> > >
> > > @@ -1098,6 +1055,7 @@ bool PipelineHandlerRkISP1::match(DeviceEnumerator *enumerator)
> > >  	mainPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
> > >  	selfPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
> > >  	stat_->bufferReady.connect(this, &PipelineHandlerRkISP1::statReady);
> > > +	isp_->frameStart.connect(this, &PipelineHandlerRkISP1::frameStart);
> > >  	param_->bufferReady.connect(this, &PipelineHandlerRkISP1::paramReady);
> > >
> > >  	/*
> > > @@ -1177,8 +1135,6 @@ void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
> > >  	if (!info)
> > >  		return;
> > >
> > > -	data->timeline_.bufferReady(buffer);
> > > -
> > >  	if (data->frame_ <= buffer->metadata().sequence)
> > >  		data->frame_ = buffer->metadata().sequence + 1;
> > >
> > > @@ -1188,6 +1144,20 @@ void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
> > >  	data->ipa_->processEvent(op);
> > >  }
> > >
> > > +void PipelineHandlerRkISP1::frameStart(uint32_t sequence)
> > > +{
> > > +	if (!activeCamera_)
> > > +		return;
> > > +
> > > +	RkISP1CameraData *data = cameraData(activeCamera_);
> > > +
> > > +	RkISP1FrameInfo *info = data->frameInfo_.find(sequence);
> > > +	if (!info || !info->paramFilled)
> > > +		LOG(RkISP1, Info)
> > > +			<< "Parameters not ready on time for frame "
> > > +			<< sequence;
> > > +}
> > > +
> > 
> > This surely is my lack of understanding, but shouldn't we worry about
> > this a param buffer queue time ?
> > 
> > >  REGISTER_PIPELINE_HANDLER(PipelineHandlerRkISP1)
> > >
> > >  } /* namespace libcamera */
> 
> -- 
> Regards,
> 
> Laurent Pinchart

Patch
diff mbox series

diff --git a/src/libcamera/pipeline/rkisp1/rkisp1.cpp b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
index c3c4b5a65e3d9afe..3662a53ac4a43fcd 100644
--- a/src/libcamera/pipeline/rkisp1/rkisp1.cpp
+++ b/src/libcamera/pipeline/rkisp1/rkisp1.cpp
@@ -40,7 +40,6 @@  namespace libcamera {
 LOG_DEFINE_CATEGORY(RkISP1)
 
 class PipelineHandlerRkISP1;
-class RkISP1ActionQueueBuffers;
 class RkISP1CameraData;
 
 enum RkISP1ActionType {
@@ -203,7 +202,6 @@  private:
 			PipelineHandler::cameraData(camera));
 	}
 
-	friend RkISP1ActionQueueBuffers;
 	friend RkISP1CameraData;
 	friend RkISP1Frames;
 
@@ -214,6 +212,7 @@  private:
 	void bufferReady(FrameBuffer *buffer);
 	void paramReady(FrameBuffer *buffer);
 	void statReady(FrameBuffer *buffer);
+	void frameStart(uint32_t sequence);
 
 	int allocateBuffers(Camera *camera);
 	int freeBuffers(Camera *camera);
@@ -348,53 +347,6 @@  RkISP1FrameInfo *RkISP1Frames::find(Request *request)
 	return nullptr;
 }
 
-class RkISP1ActionQueueBuffers : public FrameAction
-{
-public:
-	RkISP1ActionQueueBuffers(unsigned int frame, RkISP1CameraData *data,
-				 PipelineHandlerRkISP1 *pipe)
-		: FrameAction(frame, QueueBuffers), data_(data), pipe_(pipe)
-	{
-	}
-
-protected:
-	void run() override
-	{
-		RkISP1FrameInfo *info = data_->frameInfo_.find(frame());
-		if (!info)
-			LOG(RkISP1, Fatal) << "Frame not known";
-
-		/*
-		 * \todo: If parameters are not filled a better method to handle
-		 * the situation than queuing a buffer with unknown content
-		 * should be used.
-		 *
-		 * It seems excessive to keep an internal zeroed scratch
-		 * parameters buffer around as this should not happen unless the
-		 * devices is under too much load. Perhaps failing the request
-		 * and returning it to the application with an error code is
-		 * better than queue it to hardware?
-		 */
-		if (!info->paramFilled)
-			LOG(RkISP1, Error)
-				<< "Parameters not ready on time for frame "
-				<< frame();
-
-		pipe_->param_->queueBuffer(info->paramBuffer);
-		pipe_->stat_->queueBuffer(info->statBuffer);
-
-		if (info->mainPathBuffer)
-			pipe_->mainPath_.queueBuffer(info->mainPathBuffer);
-
-		if (info->selfPathBuffer)
-			pipe_->selfPath_.queueBuffer(info->selfPathBuffer);
-	}
-
-private:
-	RkISP1CameraData *data_;
-	PipelineHandlerRkISP1 *pipe_;
-};
-
 int RkISP1CameraData::loadIPA()
 {
 	ipa_ = IPAManager::createIPA(pipe_, 1, 1);
@@ -958,9 +910,14 @@  int PipelineHandlerRkISP1::queueRequestDevice(Camera *camera, Request *request)
 	op.controls = { request->controls() };
 	data->ipa_->processEvent(op);
 
-	data->timeline_.scheduleAction(std::make_unique<RkISP1ActionQueueBuffers>(data->frame_,
-										  data,
-										  this));
+	param_->queueBuffer(info->paramBuffer);
+	stat_->queueBuffer(info->statBuffer);
+
+	if (info->mainPathBuffer)
+		mainPath_.queueBuffer(info->mainPathBuffer);
+
+	if (info->selfPathBuffer)
+		selfPath_.queueBuffer(info->selfPathBuffer);
 
 	data->frame_++;
 
@@ -1098,6 +1055,7 @@  bool PipelineHandlerRkISP1::match(DeviceEnumerator *enumerator)
 	mainPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
 	selfPath_.bufferReady().connect(this, &PipelineHandlerRkISP1::bufferReady);
 	stat_->bufferReady.connect(this, &PipelineHandlerRkISP1::statReady);
+	isp_->frameStart.connect(this, &PipelineHandlerRkISP1::frameStart);
 	param_->bufferReady.connect(this, &PipelineHandlerRkISP1::paramReady);
 
 	/*
@@ -1177,8 +1135,6 @@  void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
 	if (!info)
 		return;
 
-	data->timeline_.bufferReady(buffer);
-
 	if (data->frame_ <= buffer->metadata().sequence)
 		data->frame_ = buffer->metadata().sequence + 1;
 
@@ -1188,6 +1144,20 @@  void PipelineHandlerRkISP1::statReady(FrameBuffer *buffer)
 	data->ipa_->processEvent(op);
 }
 
+void PipelineHandlerRkISP1::frameStart(uint32_t sequence)
+{
+	if (!activeCamera_)
+		return;
+
+	RkISP1CameraData *data = cameraData(activeCamera_);
+
+	RkISP1FrameInfo *info = data->frameInfo_.find(sequence);
+	if (!info || !info->paramFilled)
+		LOG(RkISP1, Info)
+			<< "Parameters not ready on time for frame "
+			<< sequence;
+}
+
 REGISTER_PIPELINE_HANDLER(PipelineHandlerRkISP1)
 
 } /* namespace libcamera */