diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/libcamera/pipeline/rcar-gen4/meson.build | 5 | ||||
-rw-r--r-- | src/libcamera/pipeline/rcar-gen4/rcar-gen4.cpp | 240 |
2 files changed, 245 insertions, 0 deletions
diff --git a/src/libcamera/pipeline/rcar-gen4/meson.build b/src/libcamera/pipeline/rcar-gen4/meson.build new file mode 100644 index 00000000..1d514677 --- /dev/null +++ b/src/libcamera/pipeline/rcar-gen4/meson.build @@ -0,0 +1,5 @@ +# SPDX-License-Identifier: CC0-1.0 + +libcamera_internal_sources += files([ + 'rcar-gen4.cpp', +]) diff --git a/src/libcamera/pipeline/rcar-gen4/rcar-gen4.cpp b/src/libcamera/pipeline/rcar-gen4/rcar-gen4.cpp new file mode 100644 index 00000000..9e5b4419 --- /dev/null +++ b/src/libcamera/pipeline/rcar-gen4/rcar-gen4.cpp @@ -0,0 +1,240 @@ +/* SPDX-License-Identifier: LGPL-2.1-or-later */ +/* + * Copyright (C) 2024, Ideas On Board Oy + * + * Pipeline handler for Renesas R-Car Gen4 + */ + +#include <memory> +#include <string> + +#include <libcamera/stream.h> + +#include "libcamera/internal/camera.h" +#include "libcamera/internal/device_enumerator.h" +#include "libcamera/internal/media_device.h" +#include "libcamera/internal/pipeline_handler.h" +#include "libcamera/internal/v4l2_subdevice.h" +#include "libcamera/internal/v4l2_videodevice.h" + +namespace libcamera { + +LOG_DEFINE_CATEGORY(RCar4) + +class RCar4CameraData : public Camera::Private +{ +public: + RCar4CameraData(PipelineHandler *pipe) + : Camera::Private(pipe) + { + } + + Stream humanVision; + Stream machineVision; +}; + +struct RCar4CameraPipe { + MediaDevice *mdev; + std::unique_ptr<V4L2Subdevice> ispCore; + std::unique_ptr<V4L2Subdevice> ispCS; + std::unique_ptr<V4L2Subdevice> csi2Rx; + std::unique_ptr<V4L2Subdevice> sensor; + std::unique_ptr<V4L2VideoDevice> params; + std::unique_ptr<V4L2VideoDevice> stats; + std::unique_ptr<V4L2VideoDevice> exposure1; + std::unique_ptr<V4L2VideoDevice> exposure2; + std::unique_ptr<V4L2VideoDevice> machineVision; + std::unique_ptr<V4L2VideoDevice> humanVision; +}; + +class PipelineHandlerRCar4 : public PipelineHandler +{ +public: + PipelineHandlerRCar4(CameraManager *manager); + + std::unique_ptr<CameraConfiguration> generateConfiguration(Camera *camera, + Span<const StreamRole> roles) override; + int configure(Camera *camera, CameraConfiguration *config) override; + + int exportFrameBuffers(Camera *camera, Stream *stream, + std::vector<std::unique_ptr<FrameBuffer>> *buffers) override; + + int start(Camera *camera, const ControlList *controls) override; + void stopDevice(Camera *camera) override; + + int queueRequestDevice(Camera *camera, Request *request) override; + + bool match(DeviceEnumerator *enumerator) override; + +private: + int createCamera(MediaDevice *mdev, std::string &pipeId); + + std::vector<RCar4CameraPipe> pipes_; +}; + +PipelineHandlerRCar4::PipelineHandlerRCar4(CameraManager *manager) + : PipelineHandler(manager) +{ +} + +std::unique_ptr<CameraConfiguration> +PipelineHandlerRCar4::generateConfiguration([[maybe_unused]] Camera *camera, + [[maybe_unused]] Span<const StreamRole> roles) +{ + return {}; +} + +int PipelineHandlerRCar4::configure([[maybe_unused]] Camera *camera, + [[maybe_unused]] CameraConfiguration *c) +{ + return 0; +} + +int PipelineHandlerRCar4::exportFrameBuffers([[maybe_unused]] Camera *camera, + [[maybe_unused]] Stream *stream, + [[maybe_unused]] std::vector<std::unique_ptr<FrameBuffer>> *buffers) +{ + return 0; +} + +int PipelineHandlerRCar4::start([[maybe_unused]] Camera *camera, + [[maybe_unused]] const ControlList *controls) +{ + return 0; +} + +void PipelineHandlerRCar4::stopDevice([[maybe_unused]] Camera *camera) +{ +} + +int PipelineHandlerRCar4::queueRequestDevice([[maybe_unused]] Camera *camera, + [[maybe_unused]] Request *request) +{ + return 0; +} + +int PipelineHandlerRCar4::createCamera(MediaDevice *mdev, std::string &pipeId) +{ + RCar4CameraPipe pipe{}; + + pipe.mdev = mdev; + pipe.ispCS = V4L2Subdevice::fromEntityName(mdev, pipeId); + if (!pipe.ispCS) { + LOG(RCar4, Error) << "Failed to find " << pipeId; + return -EINVAL; + } + + pipe.ispCore = V4L2Subdevice::fromEntityName(mdev, pipeId + " core"); + if (!pipe.ispCore) { + LOG(RCar4, Error) << "Failed to find " << pipeId + " core"; + return -EINVAL; + } + + /* Create the ISP video devices. */ + const MediaEntity *ispEntity = pipe.ispCore->entity(); + const MediaPad *pad = ispEntity->getPadByIndex(0); + const MediaPad *vdevPad = pad->links()[0]->source(); + pipe.exposure1 = V4L2VideoDevice::fromEntityName(mdev, + vdevPad->entity()->name()); + + pad = ispEntity->getPadByIndex(1); + vdevPad = pad->links()[0]->source(); + pipe.exposure2 = V4L2VideoDevice::fromEntityName(mdev, + vdevPad->entity()->name()); + + pad = ispEntity->getPadByIndex(2); + vdevPad = pad->links()[0]->source(); + pipe.params = V4L2VideoDevice::fromEntityName(mdev, + vdevPad->entity()->name()); + + pad = ispEntity->getPadByIndex(3); + vdevPad = pad->links()[0]->sink(); + pipe.stats = V4L2VideoDevice::fromEntityName(mdev, + vdevPad->entity()->name()); + + pad = ispEntity->getPadByIndex(4); + vdevPad = pad->links()[0]->sink(); + pipe.humanVision = V4L2VideoDevice::fromEntityName(mdev, + vdevPad->entity()->name()); + + pad = ispEntity->getPadByIndex(5); + vdevPad = pad->links()[0]->sink(); + pipe.machineVision = V4L2VideoDevice::fromEntityName(mdev, + vdevPad->entity()->name()); + + if (!pipe.exposure1 || !pipe.exposure2 || !pipe.params || !pipe.stats || + !pipe.humanVision || !pipe.machineVision) { + LOG(RCar4, Error) << "Unable to find ISP components"; + return -ENODEV; + } + + /* + * Walk the ispCS sink link to get to the csi-2 Rx and from + * there create a camera. + */ + const MediaEntity *entity = pipe.ispCS->entity(); + pad = entity->getPadByIndex(0); + const MediaPad *source = pad->links()[0]->source(); + pipe.csi2Rx = V4L2Subdevice::fromEntityName(mdev, source->entity()->name()); + if (!pipe.csi2Rx) { + LOG(RCar4, Error) << "Failed to find CSI-2 Rx entity"; + return -EINVAL; + } + + entity = pipe.csi2Rx->entity(); + pad = entity->getPadByIndex(0); + source = pad->links()[0]->source(); + pipe.sensor = V4L2Subdevice::fromEntityName(mdev, source->entity()->name()); + if (!pipe.sensor) { + LOG(RCar4, Error) << "Failed to find sensor entity"; + return -EINVAL; + } + + std::unique_ptr<RCar4CameraData> data = + std::make_unique<RCar4CameraData>(this); + + std::set<Stream *> streams{ + &data->humanVision, + &data->machineVision, + }; + + const std::string id = pipe.sensor->entity()->name(); + std::shared_ptr<Camera> camera = + Camera::create(std::move(data), id, streams); + registerCamera(std::move(camera)); + + return 0; +} + +bool PipelineHandlerRCar4::match(DeviceEnumerator *enumerator) +{ + DeviceMatch dm("rcar_vin"); + + MediaDevice *media = acquireMediaDevice(enumerator, dm); + if (!media) + return false; + + /* + * Walk the entities in the media graph to identify the required + * components. + */ + for (const MediaEntity *entity : media->entities()) { + /* Identify the ISP core entries (part of the same media graph). */ + if (entity->name().substr(0, 8) == "rcar_isp" && + entity->name().rfind("core") != std::string::npos) { + /* + * Isolate the unit address that identifies one ISP + * instance. pipeId will look like + * 'rcar_isp fed00000.isp'. + */ + std::string pipeId = entity->name().substr(0, 21); + createCamera(media, pipeId); + } + } + + return false; +} + +REGISTER_PIPELINE_HANDLER(PipelineHandlerRCar4, "rcar-gen4") + +} /* namespace libcamera */ |