The buffer index is a V4L2 concept that will be hidden from users with the introduction of a new FrameBuffer class. In preparation for this, remove the index from log messages. Keep and move one debug log message where the index is available as the V4L2 buffer is being dequeued for the video device and it's useful when debugging. Signed-off-by: Niklas Söderlund <niklas.soderlund@ragnatech.se> Reviewed-by: Jacopo Mondi <jacopo@jmondi.org> Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
198 lines
4.3 KiB
C++
198 lines
4.3 KiB
C++
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* Copyright (C) 2019, Google Inc.
|
|
*
|
|
* libcamera V4L2 API tests
|
|
*
|
|
* Validate the function of exporting buffers from a V4L2VideoDevice and
|
|
* the ability to import them to another V4L2VideoDevice instance.
|
|
* Ensure that the Buffers can successfully be queued and dequeued
|
|
* between both devices.
|
|
*/
|
|
|
|
#include <iostream>
|
|
|
|
#include <libcamera/buffer.h>
|
|
#include <libcamera/event_dispatcher.h>
|
|
#include <libcamera/timer.h>
|
|
|
|
#include "thread.h"
|
|
#include "v4l2_videodevice_test.h"
|
|
|
|
class BufferSharingTest : public V4L2VideoDeviceTest
|
|
{
|
|
public:
|
|
BufferSharingTest()
|
|
: V4L2VideoDeviceTest("vivid", "vivid-000-vid-cap"),
|
|
output_(nullptr), framesCaptured_(0), framesOutput_(0) {}
|
|
|
|
protected:
|
|
int init()
|
|
{
|
|
int ret = V4L2VideoDeviceTest::init();
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* media_ already represents VIVID */
|
|
MediaEntity *entity = media_->getEntityByName("vivid-000-vid-out");
|
|
if (!entity)
|
|
return TestSkip;
|
|
|
|
output_ = new V4L2VideoDevice(entity);
|
|
if (!output_) {
|
|
std::cout << "Failed to create output device" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
ret = output_->open();
|
|
if (ret) {
|
|
std::cout << "Failed to open output device" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
V4L2DeviceFormat format = {};
|
|
|
|
ret = capture_->getFormat(&format);
|
|
if (ret) {
|
|
std::cout << "Failed to get capture format" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
format.size.width = 320;
|
|
format.size.height = 180;
|
|
|
|
ret = capture_->setFormat(&format);
|
|
if (ret) {
|
|
std::cout << "Failed to set capture format" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
ret = output_->setFormat(&format);
|
|
if (ret) {
|
|
std::cout << "Failed to set output format" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
pool_.createBuffers(bufferCount);
|
|
|
|
ret = capture_->exportBuffers(&pool_);
|
|
if (ret) {
|
|
std::cout << "Failed to export buffers" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
ret = output_->importBuffers(&pool_);
|
|
if (ret) {
|
|
std::cout << "Failed to import buffers" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void captureBufferReady(Buffer *buffer)
|
|
{
|
|
std::cout << "Received capture buffer" << std::endl;
|
|
|
|
if (buffer->status() != Buffer::BufferSuccess)
|
|
return;
|
|
|
|
output_->queueBuffer(buffer);
|
|
framesCaptured_++;
|
|
}
|
|
|
|
void outputBufferReady(Buffer *buffer)
|
|
{
|
|
std::cout << "Received output buffer" << std::endl;
|
|
|
|
if (buffer->status() != Buffer::BufferSuccess)
|
|
return;
|
|
|
|
capture_->queueBuffer(buffer);
|
|
framesOutput_++;
|
|
}
|
|
|
|
int run()
|
|
{
|
|
EventDispatcher *dispatcher = Thread::current()->eventDispatcher();
|
|
Timer timeout;
|
|
int ret;
|
|
|
|
capture_->bufferReady.connect(this, &BufferSharingTest::captureBufferReady);
|
|
output_->bufferReady.connect(this, &BufferSharingTest::outputBufferReady);
|
|
|
|
std::vector<std::unique_ptr<Buffer>> buffers;
|
|
buffers = capture_->queueAllBuffers();
|
|
if (buffers.empty())
|
|
return TestFail;
|
|
|
|
ret = capture_->streamOn();
|
|
if (ret) {
|
|
std::cout << "Failed to start streaming on the capture device" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
ret = output_->streamOn();
|
|
if (ret) {
|
|
std::cout << "Failed to start streaming on the output device" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
timeout.start(10000);
|
|
while (timeout.isRunning()) {
|
|
dispatcher->processEvents();
|
|
if (framesCaptured_ > 30 && framesOutput_ > 30)
|
|
break;
|
|
}
|
|
|
|
if ((framesCaptured_ < 1) || (framesOutput_ < 1)) {
|
|
std::cout << "Failed to process any frames within timeout." << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
if ((framesCaptured_ < 30) || (framesOutput_ < 30)) {
|
|
std::cout << "Failed to process 30 frames within timeout." << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
ret = capture_->streamOff();
|
|
if (ret) {
|
|
std::cout << "Failed to stop streaming on the capture device" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
ret = output_->streamOff();
|
|
if (ret) {
|
|
std::cout << "Failed to stop streaming on the output device" << std::endl;
|
|
return TestFail;
|
|
}
|
|
|
|
return TestPass;
|
|
}
|
|
|
|
void cleanup()
|
|
{
|
|
std::cout
|
|
<< "Captured " << framesCaptured_ << " frames and "
|
|
<< "output " << framesOutput_ << " frames"
|
|
<< std::endl;
|
|
|
|
output_->streamOff();
|
|
output_->releaseBuffers();
|
|
output_->close();
|
|
|
|
delete output_;
|
|
|
|
V4L2VideoDeviceTest::cleanup();
|
|
}
|
|
|
|
private:
|
|
const unsigned int bufferCount = 4;
|
|
|
|
V4L2VideoDevice *output_;
|
|
|
|
unsigned int framesCaptured_;
|
|
unsigned int framesOutput_;
|
|
};
|
|
|
|
TEST_REGISTER(BufferSharingTest);
|