[PATCH 04/12] libcamera: delayed_controls: Update unit tests to expected semantics

Stefan Klug stefan.klug at ideasonboard.com
Wed Mar 13 11:56:36 CET 2024


This patches changes the tests in a way DelayedControls is expected to
work. It does not require a interface change in DelayedControls.

The changes are:
- It is expected that for every apply() there was a previous push().
- When a control is pushed for frame 0, that request is not lost, but
  applied as soon as possible

Signed-off-by: Stefan Klug <stefan.klug at ideasonboard.com>
---
 test/delayed_controls.cpp | 203 +++++++++++++++++++++++++++++---------
 1 file changed, 156 insertions(+), 47 deletions(-)

diff --git a/test/delayed_controls.cpp b/test/delayed_controls.cpp
index a8ce9828..e0634fa1 100644
--- a/test/delayed_controls.cpp
+++ b/test/delayed_controls.cpp
@@ -84,11 +84,8 @@ protected:
 		dev_->setControls(&ctrls);
 		delayed->reset();
 
-		/* Trigger the first frame start event */
-		delayed->applyControls(0);
-
 		/* Test control without delay are set at once. */
-		for (unsigned int i = 1; i < 100; i++) {
+		for (unsigned int i = 0; i < 100; i++) {
 			int32_t value = 100 + i;
 
 			ctrls.set(V4L2_CID_BRIGHTNESS, value);
@@ -121,25 +118,30 @@ protected:
 		ControlList ctrls;
 
 		/* Reset control to value that will be first in test. */
-		int32_t expected = 4;
-		ctrls.set(V4L2_CID_BRIGHTNESS, expected);
+		int32_t initial = 4;
+		ctrls.set(V4L2_CID_BRIGHTNESS, initial);
 		dev_->setControls(&ctrls);
 		delayed->reset();
 
-		/* Trigger the first frame start event */
-		delayed->applyControls(0);
+		/* Push a request for frame 0 */
+		ctrls.set(V4L2_CID_BRIGHTNESS, 10);
+		delayed->push(ctrls);
 
 		/* Test single control with delay. */
-		for (unsigned int i = 1; i < 100; i++) {
+		for (unsigned int i = 0; i < 100; i++) {
 			int32_t value = 10 + i;
+			int32_t expected = i < 1 ? initial : value;
 
-			ctrls.set(V4L2_CID_BRIGHTNESS, value);
+			/* push the request for frame i+1 */
+			ctrls.set(V4L2_CID_BRIGHTNESS, value + 1);
 			delayed->push(ctrls);
 
 			delayed->applyControls(i);
 
 			ControlList result = delayed->get(i);
 			int32_t brightness = result.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
+			ControlList ctrlsV4L = dev_->getControls({ V4L2_CID_BRIGHTNESS });
+			int32_t brightnessV4L = ctrlsV4L.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
 			if (brightness != expected) {
 				cerr << "Failed single control with delay"
 				     << " frame " << i
@@ -149,7 +151,72 @@ protected:
 				return TestFail;
 			}
 
-			expected = value;
+			if (i > 0 && brightnessV4L != value + 1) {
+				cerr << "Failed single control with delay"
+				     << " frame " << i
+				     << " expected V4L " << value + 1
+				     << " got " << brightnessV4L
+				     << endl;
+				return TestFail;
+			}
+		}
+
+		return TestPass;
+	}
+
+	/* This fails on the old delayed controls implementation */
+	int singleControlWithDelayStartUp()
+	{
+		std::unordered_map<uint32_t, DelayedControls::ControlParams> delays = {
+			{ V4L2_CID_BRIGHTNESS, { 1, false } },
+		};
+		std::unique_ptr<DelayedControls> delayed =
+			std::make_unique<DelayedControls>(dev_.get(), delays);
+		ControlList ctrls;
+
+		/* Reset control to value that will be first in test. */
+		int32_t initial = 4;
+		ctrls.set(V4L2_CID_BRIGHTNESS, initial);
+		dev_->setControls(&ctrls);
+		delayed->reset();
+
+		/* push a request for frame 0 */
+		ctrls.set(V4L2_CID_BRIGHTNESS, 10);
+		delayed->push(ctrls);
+
+		/* Test single control with delay. */
+		for (unsigned int i = 0; i < 100; i++) {
+			int32_t value = 10 + i;
+			int32_t expected = i < 1 ? initial : value;
+
+			/* push the request for frame i+1 */
+			ctrls.set(V4L2_CID_BRIGHTNESS, value + 1);
+			delayed->push(ctrls);
+
+			delayed->applyControls(i);
+
+			ControlList result = delayed->get(i);
+			int32_t brightness = result.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
+			ControlList ctrlsV4L = dev_->getControls({ V4L2_CID_BRIGHTNESS });
+			int32_t brightnessV4L = ctrlsV4L.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
+
+			if (brightness != expected) {
+				cerr << "Failed single control with delay start up"
+				     << " frame " << i
+				     << " expected " << expected
+				     << " got " << brightness
+				     << endl;
+				return TestFail;
+			}
+
+			if (i > 0 && brightnessV4L != value + 1) {
+				cerr << "Failed single control with delay start up"
+				     << " frame " << i
+				     << " expected V4L " << value + 1
+				     << " got " << brightnessV4L
+				     << endl;
+				return TestFail;
+			}
 		}
 
 		return TestPass;
@@ -157,7 +224,7 @@ protected:
 
 	int dualControlsWithDelay()
 	{
-		static const unsigned int maxDelay = 2;
+		static const int maxDelay = 2;
 
 		std::unordered_map<uint32_t, DelayedControls::ControlParams> delays = {
 			{ V4L2_CID_BRIGHTNESS, { 1, false } },
@@ -174,15 +241,21 @@ protected:
 		dev_->setControls(&ctrls);
 		delayed->reset();
 
-		/* Trigger the first frame start event */
-		delayed->applyControls(0);
+		/* push two requests into the queue */
+		ctrls.set(V4L2_CID_BRIGHTNESS, 10);
+		ctrls.set(V4L2_CID_CONTRAST, 10);
+		delayed->push(ctrls);
+		ctrls.set(V4L2_CID_BRIGHTNESS, 11);
+		ctrls.set(V4L2_CID_CONTRAST, 11);
+		delayed->push(ctrls);
 
 		/* Test dual control with delay. */
-		for (unsigned int i = 1; i < 100; i++) {
+		for (unsigned int i = 0; i < 100; i++) {
 			int32_t value = 10 + i;
 
-			ctrls.set(V4L2_CID_BRIGHTNESS, value);
-			ctrls.set(V4L2_CID_CONTRAST, value + 1);
+			/* we are pushing 2 frames ahead */
+			ctrls.set(V4L2_CID_BRIGHTNESS, value + 2);
+			ctrls.set(V4L2_CID_CONTRAST, value + 2);
 			delayed->push(ctrls);
 
 			delayed->applyControls(i);
@@ -190,17 +263,32 @@ protected:
 			ControlList result = delayed->get(i);
 			int32_t brightness = result.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
 			int32_t contrast = result.get(V4L2_CID_CONTRAST).get<int32_t>();
-			if (brightness != expected || contrast != expected + 1) {
-				cerr << "Failed dual controls"
-				     << " frame " << i
-				     << " brightness " << brightness
-				     << " contrast " << contrast
-				     << " expected " << expected
-				     << endl;
-				return TestFail;
-			}
 
-			expected = i < maxDelay ? expected : value - 1;
+			ControlList ctrlsV4L = dev_->getControls({ V4L2_CID_BRIGHTNESS, V4L2_CID_CONTRAST });
+			int32_t brightnessV4L = ctrlsV4L.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
+			int32_t contrastV4L = ctrlsV4L.get(V4L2_CID_CONTRAST).get<int32_t>();
+
+			if (i > maxDelay) {
+				if (brightness != value || contrast != value) {
+					cerr << "Failed dual controls"
+					     << " frame " << i
+					     << " brightness " << brightness
+					     << " contrast " << contrast
+					     << " expected " << value
+					     << endl;
+					return TestFail;
+				}
+				if (brightnessV4L != value + 1 || contrastV4L != value + maxDelay) {
+					cerr << "Failed dual controls"
+					     << " frame " << i
+					     << " brightnessV4L " << brightnessV4L
+					     << " expected " << value + 1
+					     << " contrastV4L " << contrastV4L
+					     << " expected " << value + maxDelay
+					     << endl;
+					return TestFail;
+				}
+			}
 		}
 
 		return TestPass;
@@ -208,7 +296,7 @@ protected:
 
 	int dualControlsMultiQueue()
 	{
-		static const unsigned int maxDelay = 2;
+		static const int maxDelay = 2;
 
 		std::unordered_map<uint32_t, DelayedControls::ControlParams> delays = {
 			{ V4L2_CID_BRIGHTNESS, { 1, false } },
@@ -218,22 +306,19 @@ protected:
 			std::make_unique<DelayedControls>(dev_.get(), delays);
 		ControlList ctrls;
 
-		/* Reset control to value that will be first two frames in test. */
-		int32_t expected = 100;
-		ctrls.set(V4L2_CID_BRIGHTNESS, expected);
-		ctrls.set(V4L2_CID_CONTRAST, expected);
+		/* Reset control to a value that will be first two frames in test. */
+		int32_t initial = 100;
+		ctrls.set(V4L2_CID_BRIGHTNESS, initial);
+		ctrls.set(V4L2_CID_CONTRAST, initial);
 		dev_->setControls(&ctrls);
 		delayed->reset();
 
-		/* Trigger the first frame start event */
-		delayed->applyControls(0);
-
 		/*
-		 * Queue all controls before any fake frame start. Note we
+		 * Queue all controls before applyControls(). Note we
 		 * can't queue up more then the delayed controls history size
-		 * which is 16. Where one spot is used by the reset control.
+		 * which is 16.
 		 */
-		for (unsigned int i = 0; i < 15; i++) {
+		for (unsigned int i = 0; i < 14; i++) {
 			int32_t value = 10 + i;
 
 			ctrls.set(V4L2_CID_BRIGHTNESS, value);
@@ -241,9 +326,9 @@ protected:
 			delayed->push(ctrls);
 		}
 
-		/* Process all queued controls. */
-		for (unsigned int i = 1; i < 16; i++) {
-			int32_t value = 10 + i - 1;
+		/* Process 2 frames less than queued, so that the V4L controls are correct */
+		for (unsigned int i = 0; i < 12; i++) {
+			int32_t expected = i < maxDelay ? initial : 10 + i;
 
 			delayed->applyControls(i);
 
@@ -251,6 +336,11 @@ protected:
 
 			int32_t brightness = result.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
 			int32_t contrast = result.get(V4L2_CID_CONTRAST).get<int32_t>();
+
+			ControlList ctrlsV4L = dev_->getControls({ V4L2_CID_BRIGHTNESS, V4L2_CID_CONTRAST });
+			int32_t brightnessV4L = ctrlsV4L.get(V4L2_CID_BRIGHTNESS).get<int32_t>();
+			int32_t contrastV4L = ctrlsV4L.get(V4L2_CID_CONTRAST).get<int32_t>();
+
 			if (brightness != expected || contrast != expected) {
 				cerr << "Failed multi queue"
 				     << " frame " << i
@@ -261,7 +351,17 @@ protected:
 				return TestFail;
 			}
 
-			expected = i < maxDelay ? expected : value - 1;
+			/* Check v4l values after they've settled */
+			if (i >= maxDelay && (brightnessV4L != expected + 1 || contrastV4L != expected + maxDelay)) {
+				cerr << "Failed multi queue"
+				     << " frame " << i
+				     << " brightnessV4L " << brightnessV4L
+				     << " expected " << expected + 1
+				     << " contrastV4L " << contrastV4L
+				     << " expected " << expected + maxDelay
+				     << endl;
+				return TestFail;
+			}
 		}
 
 		return TestPass;
@@ -269,27 +369,36 @@ protected:
 
 	int run() override
 	{
-		int ret;
+		int ret = 0;
+		bool failed = false;
 
 		/* Test single control without delay. */
 		ret = singleControlNoDelay();
 		if (ret)
-			return ret;
+			failed = true;
 
 		/* Test single control with delay. */
 		ret = singleControlWithDelay();
 		if (ret)
-			return ret;
+			failed = true;
+
+		/* Test single control with delay. */
+		ret = singleControlWithDelayStartUp();
+		if (ret)
+			failed = true;
 
 		/* Test dual controls with different delays. */
 		ret = dualControlsWithDelay();
 		if (ret)
-			return ret;
+			failed = true;
 
 		/* Test control values produced faster than consumed. */
 		ret = dualControlsMultiQueue();
 		if (ret)
-			return ret;
+			failed = true;
+
+		if (failed)
+			return TestFail;
 
 		return TestPass;
 	}
-- 
2.40.1



More information about the libcamera-devel mailing list