libcamera: pipeline: Add test pattern for VirtualPipelineHandler

Add a test pattern generator class hierarchy for the Virtual
pipeline handler.

Implement two types of test patterns: color bars and diagonal lines
generator and use them in the Virtual pipeline handler.

A shifting mechanism is enabled. For each frame, the image is shifted to
the left by 1 pixel. It drops FPS though.

Add a dependency for libyuv to the build system to generate images
in NV12 format from the test pattern.

Signed-off-by: Konami Shu <konamiz@google.com>
Co-developed-by: Harvey Yang <chenghaoyang@chromium.org>
Signed-off-by: Harvey Yang <chenghaoyang@chromium.org>
Co-developed-by: Yunke Cao <yunkec@chromium.org>
Signed-off-by: Yunke Cao <yunkec@chromium.org>
Co-developed-by: Tomasz Figa <tfiga@chromium.org>
Signed-off-by: Tomasz Figa <tfiga@chromium.org>
Reviewed-by: Jacopo Mondi <jacopo.mondi@ideasonboard.com>
Reviewed-by: Kieran Bingham <kieran.bingham@ideasonboard.com>
Signed-off-by: Kieran Bingham <kieran.bingham@ideasonboard.com>
This commit is contained in:
Harvey Yang 2024-10-22 07:43:40 +00:00 committed by Kieran Bingham
parent 3a884ebfe4
commit eeaa7de21b
8 changed files with 282 additions and 22 deletions

View file

@ -15,25 +15,6 @@ foreach dep : android_deps
endif
endforeach
libyuv_dep = dependency('libyuv', required : false)
# Fallback to a subproject if libyuv isn't found, as it's typically not
# provided by distributions.
if not libyuv_dep.found()
cmake = import('cmake')
libyuv_vars = cmake.subproject_options()
libyuv_vars.add_cmake_defines({'CMAKE_POSITION_INDEPENDENT_CODE': 'ON'})
libyuv_vars.set_override_option('cpp_std', 'c++17')
libyuv_vars.append_compile_args('cpp',
'-Wno-sign-compare',
'-Wno-unused-variable',
'-Wno-unused-parameter')
libyuv_vars.append_link_args('-ljpeg')
libyuv = cmake.subproject('libyuv', options : libyuv_vars)
libyuv_dep = libyuv.dependency('yuv')
endif
android_deps += [libyuv_dep]
android_hal_sources = files([

View file

@ -0,0 +1,29 @@
/* SPDX-License-Identifier: LGPL-2.1-or-later */
/*
* Copyright (C) 2024, Google Inc.
*
* Virtual cameras helper to generate frames
*/
#pragma once
#include <libcamera/framebuffer.h>
#include <libcamera/geometry.h>
namespace libcamera {
class FrameGenerator
{
public:
virtual ~FrameGenerator() = default;
virtual void configure(const Size &size) = 0;
virtual int generateFrame(const Size &size,
const FrameBuffer *buffer) = 0;
protected:
FrameGenerator() {}
};
} /* namespace libcamera */

View file

@ -1,5 +1,8 @@
# SPDX-License-Identifier: CC0-1.0
libcamera_internal_sources += files([
'test_pattern_generator.cpp',
'virtual.cpp',
])
libcamera_deps += [libyuv_dep]

View file

@ -0,0 +1,136 @@
/* SPDX-License-Identifier: LGPL-2.1-or-later */
/*
* Copyright (C) 2024, Google Inc.
*
* Derived class of FrameGenerator for generating test patterns
*/
#include "test_pattern_generator.h"
#include <libcamera/base/log.h>
#include "libcamera/internal/mapped_framebuffer.h"
#include <libyuv/convert_from_argb.h>
namespace libcamera {
LOG_DECLARE_CATEGORY(Virtual)
static const unsigned int kARGBSize = 4;
int TestPatternGenerator::generateFrame(const Size &size,
const FrameBuffer *buffer)
{
MappedFrameBuffer mappedFrameBuffer(buffer,
MappedFrameBuffer::MapFlag::Write);
auto planes = mappedFrameBuffer.planes();
shiftLeft(size);
/* Convert the template_ to the frame buffer */
int ret = libyuv::ARGBToNV12(template_.get(), size.width * kARGBSize,
planes[0].begin(), size.width,
planes[1].begin(), size.width,
size.width, size.height);
if (ret != 0)
LOG(Virtual, Error) << "ARGBToNV12() failed with " << ret;
return ret;
}
void TestPatternGenerator::shiftLeft(const Size &size)
{
/* Store the first column temporarily */
auto firstColumn = std::make_unique<uint8_t[]>(size.height * kARGBSize);
for (size_t h = 0; h < size.height; h++) {
unsigned int index = h * size.width * kARGBSize;
unsigned int index1 = h * kARGBSize;
firstColumn[index1] = template_[index];
firstColumn[index1 + 1] = template_[index + 1];
firstColumn[index1 + 2] = template_[index + 2];
firstColumn[index1 + 3] = 0x00;
}
/* Overwrite template_ */
uint8_t *buf = template_.get();
for (size_t h = 0; h < size.height; h++) {
for (size_t w = 0; w < size.width - 1; w++) {
/* Overwrite with the pixel on the right */
unsigned int index = (h * size.width + w + 1) * kARGBSize;
*buf++ = template_[index]; /* B */
*buf++ = template_[index + 1]; /* G */
*buf++ = template_[index + 2]; /* R */
*buf++ = 0x00; /* A */
}
/* Overwrite the new last column with the original first column */
unsigned int index1 = h * kARGBSize;
*buf++ = firstColumn[index1]; /* B */
*buf++ = firstColumn[index1 + 1]; /* G */
*buf++ = firstColumn[index1 + 2]; /* R */
*buf++ = 0x00; /* A */
}
}
void ColorBarsGenerator::configure(const Size &size)
{
constexpr uint8_t kColorBar[8][3] = {
/* R, G, B */
{ 0xff, 0xff, 0xff }, /* White */
{ 0xff, 0xff, 0x00 }, /* Yellow */
{ 0x00, 0xff, 0xff }, /* Cyan */
{ 0x00, 0xff, 0x00 }, /* Green */
{ 0xff, 0x00, 0xff }, /* Magenta */
{ 0xff, 0x00, 0x00 }, /* Red */
{ 0x00, 0x00, 0xff }, /* Blue */
{ 0x00, 0x00, 0x00 }, /* Black */
};
template_ = std::make_unique<uint8_t[]>(
size.width * size.height * kARGBSize);
unsigned int colorBarWidth = size.width / std::size(kColorBar);
uint8_t *buf = template_.get();
for (size_t h = 0; h < size.height; h++) {
for (size_t w = 0; w < size.width; w++) {
/* Repeat when the width is exceed */
unsigned int index = (w / colorBarWidth) % std::size(kColorBar);
*buf++ = kColorBar[index][2]; /* B */
*buf++ = kColorBar[index][1]; /* G */
*buf++ = kColorBar[index][0]; /* R */
*buf++ = 0x00; /* A */
}
}
}
void DiagonalLinesGenerator::configure(const Size &size)
{
constexpr uint8_t kColorBar[2][3] = {
/* R, G, B */
{ 0xff, 0xff, 0xff }, /* White */
{ 0x00, 0x00, 0x00 }, /* Black */
};
template_ = std::make_unique<uint8_t[]>(
size.width * size.height * kARGBSize);
unsigned int lineWidth = size.width / 10;
uint8_t *buf = template_.get();
for (size_t h = 0; h < size.height; h++) {
for (size_t w = 0; w < size.width; w++) {
/* Repeat when the width is exceed */
int index = ((w + h) / lineWidth) % 2;
*buf++ = kColorBar[index][2]; /* B */
*buf++ = kColorBar[index][1]; /* G */
*buf++ = kColorBar[index][0]; /* R */
*buf++ = 0x00; /* A */
}
}
}
} /* namespace libcamera */

View file

@ -0,0 +1,52 @@
/* SPDX-License-Identifier: LGPL-2.1-or-later */
/*
* Copyright (C) 2024, Google Inc.
*
* Derived class of FrameGenerator for generating test patterns
*/
#pragma once
#include <memory>
#include <libcamera/framebuffer.h>
#include <libcamera/geometry.h>
#include "frame_generator.h"
namespace libcamera {
enum class TestPattern : char {
ColorBars = 0,
DiagonalLines = 1,
};
class TestPatternGenerator : public FrameGenerator
{
public:
int generateFrame(const Size &size, const FrameBuffer *buffer) override;
protected:
/* Buffer of test pattern template */
std::unique_ptr<uint8_t[]> template_;
private:
/* Shift the buffer by 1 pixel left each frame */
void shiftLeft(const Size &size);
};
class ColorBarsGenerator : public TestPatternGenerator
{
public:
/* Generate a template buffer of the color bar test pattern. */
void configure(const Size &size) override;
};
class DiagonalLinesGenerator : public TestPatternGenerator
{
public:
/* Generate a template buffer of the diagonal lines test pattern. */
void configure(const Size &size) override;
};
} /* namespace libcamera */

View file

@ -34,6 +34,7 @@
#include "libcamera/internal/camera.h"
#include "libcamera/internal/dma_buf_allocator.h"
#include "libcamera/internal/formats.h"
#include "libcamera/internal/framebuffer.h"
#include "libcamera/internal/pipeline_handler.h"
namespace libcamera {
@ -94,6 +95,8 @@ private:
return static_cast<VirtualCameraData *>(camera->_d());
}
void initFrameGenerator(Camera *camera);
DmaBufAllocator dmaBufAllocator_;
bool resetCreated_ = false;
@ -241,8 +244,10 @@ int PipelineHandlerVirtual::configure(Camera *camera,
CameraConfiguration *config)
{
VirtualCameraData *data = cameraData(camera);
for (auto [i, c] : utils::enumerate(*config))
for (auto [i, c] : utils::enumerate(*config)) {
c.setStream(&data->streamConfigs_[i].stream);
data->streamConfigs_[i].frameGenerator->configure(c.size);
}
return 0;
}
@ -278,8 +283,24 @@ void PipelineHandlerVirtual::stopDevice([[maybe_unused]] Camera *camera)
int PipelineHandlerVirtual::queueRequestDevice([[maybe_unused]] Camera *camera,
Request *request)
{
for (auto it : request->buffers())
completeBuffer(request, it.second);
VirtualCameraData *data = cameraData(camera);
for (auto const &[stream, buffer] : request->buffers()) {
bool found = false;
/* map buffer and fill test patterns */
for (auto &streamConfig : data->streamConfigs_) {
if (stream == &streamConfig.stream) {
found = true;
if (streamConfig.frameGenerator->generateFrame(
stream->configuration().size, buffer))
buffer->_d()->cancel();
completeBuffer(request, buffer);
break;
}
}
ASSERT(found);
}
request->metadata().set(controls::SensorTimestamp, currentTimestamp());
completeRequest(request);
@ -325,6 +346,9 @@ bool PipelineHandlerVirtual::match([[maybe_unused]] DeviceEnumerator *enumerator
const std::string id = "Virtual0";
std::shared_ptr<Camera> camera = Camera::create(std::move(data), id, streams);
initFrameGenerator(camera.get());
registerCamera(std::move(camera));
resetCreated_ = true;
@ -332,6 +356,17 @@ bool PipelineHandlerVirtual::match([[maybe_unused]] DeviceEnumerator *enumerator
return true;
}
void PipelineHandlerVirtual::initFrameGenerator(Camera *camera)
{
auto data = cameraData(camera);
for (auto &streamConfig : data->streamConfigs_) {
if (data->testPattern_ == TestPattern::DiagonalLines)
streamConfig.frameGenerator = std::make_unique<DiagonalLinesGenerator>();
else
streamConfig.frameGenerator = std::make_unique<ColorBarsGenerator>();
}
}
REGISTER_PIPELINE_HANDLER(PipelineHandlerVirtual, "virtual")
} /* namespace libcamera */

View file

@ -15,6 +15,8 @@
#include "libcamera/internal/camera.h"
#include "libcamera/internal/pipeline_handler.h"
#include "test_pattern_generator.h"
namespace libcamera {
class VirtualCameraData : public Camera::Private
@ -28,6 +30,7 @@ public:
};
struct StreamConfig {
Stream stream;
std::unique_ptr<FrameGenerator> frameGenerator;
};
VirtualCameraData(PipelineHandler *pipe,
@ -35,6 +38,8 @@ public:
~VirtualCameraData() = default;
TestPattern testPattern_ = TestPattern::ColorBars;
const std::vector<Resolution> supportedResolutions_;
Size maxResolutionSize_;
Size minResolutionSize_;

View file

@ -27,6 +27,25 @@ else
ipa_sign_module = false
endif
libyuv_dep = dependency('libyuv', required : false)
# Fallback to a subproject if libyuv isn't found, as it's typically not
# provided by distributions.
if not libyuv_dep.found()
cmake = import('cmake')
libyuv_vars = cmake.subproject_options()
libyuv_vars.add_cmake_defines({'CMAKE_POSITION_INDEPENDENT_CODE': 'ON'})
libyuv_vars.set_override_option('cpp_std', 'c++17')
libyuv_vars.append_compile_args('cpp',
'-Wno-sign-compare',
'-Wno-unused-variable',
'-Wno-unused-parameter')
libyuv_vars.append_link_args('-ljpeg')
libyuv = cmake.subproject('libyuv', options : libyuv_vars)
libyuv_dep = libyuv.dependency('yuv')
endif
# libcamera must be built first as a dependency to the other components.
subdir('libcamera')