summaryrefslogtreecommitdiff
path: root/src/cam
AgeCommit message (Expand)Author
2020-01-12libcamera: buffer: Switch from Plane to FrameBuffer::PlaneNiklas Söderlund
2019-12-16libcamera: Remove buffer index from loggingNiklas Söderlund
2019-11-26cam: options: Fix unitialized variable warningLaurent Pinchart
2019-11-20cam: Store camera as shared pointer everywhereNiklas Söderlund
2019-11-20cam: BufferWriter: Use the libcamera namespaceNiklas Söderlund
2019-11-19libcamera: camera: Remove explicit stream to buffer map in requestCompleted s...Niklas Söderlund
2019-11-08libcamera: Remove unneeded semicolonsLaurent Pinchart
2019-10-26cam: capture: remove unused local variablePaul Elder
2019-10-23libcamera: Standardise on C compatibility headersLaurent Pinchart
2019-09-14libcamera: Switch to the std::chrono APILaurent Pinchart
2019-08-19libcamera: camera_manager: Construct CameraManager instances manuallyLaurent Pinchart
2019-07-14libcamera: buffer: Add an accessor to the BufferMemoryLaurent Pinchart
2019-07-14libcamera: stream: Shorten access to the bufferPoolJacopo Mondi
2019-07-14libcamera: Stop using Stream::bufferPool to get the number of buffersLaurent Pinchart
2019-07-14libcamera: buffer: Split memory information to BufferMemoryLaurent Pinchart
2019-06-25cam: capture: Stop stream when queueRequest() failsHelen Koike
2019-06-19cam: Support base 16 and base 8 when parsing integer optionsLaurent Pinchart
2019-06-19cam: Add --info option to print information about stream(s)Niklas Söderlund
2019-06-19cam: Validate camera configurationNiklas Söderlund
2019-06-19cam: Move camera configuration preparation to CamAppNiklas Söderlund
2019-06-18cam: Allow selecting cameras by indexLaurent Pinchart
2019-06-10cam: Fix cam --help crashNiklas Söderlund
2019-05-25cam: Add CamApp classNiklas Söderlund
2019-05-25cam: capture: Break out capture to a new classNiklas Söderlund
2019-05-23meson: Create and use a dependency for libcamera and its headersLaurent Pinchart
2019-05-23cam: Rename conf variable referring to command line option to optLaurent Pinchart
2019-05-23libcamera: camera: Add a validation API to the CameraConfiguration classLaurent Pinchart
2019-05-23libcamera: camera: Return a pointer from generateConfiguration()Laurent Pinchart
2019-05-23libcamera: Refactor the camera configuration storage and APILaurent Pinchart
2019-05-23libcamera: Use stream roles directly instead of StreamUsageLaurent Pinchart
2019-05-23libcamera: camera: Rename configureStreams() and streamConfiguration()Laurent Pinchart
2019-04-30libcamera: Use the Size class through libcameraLaurent Pinchart
2019-04-26cam: options: Fix string concatenationLaurent Pinchart
2019-04-26cam: options: Don't initialise variable-length arraysLaurent Pinchart
2019-04-18Install the cam and qcam utilitiesLaurent Pinchart
2019-04-09cam: Allow cameras with more than one streamNiklas Söderlund
2019-04-09cam: Extend request completion handler to deal with multiple streamsNiklas Söderlund
2019-04-09cam: Add support to specify multiple stream configurations with rolesNiklas Söderlund
2019-04-09cam: Extend BufferWriter to include a stream name in file pathNiklas Söderlund
2019-04-09cam: Rename --format to --streamNiklas Söderlund
2019-04-09libcamera: Switch to CameraConfigurationNiklas Söderlund
2019-04-05libcamera: camera: Add support for stream usagesNiklas Söderlund
2019-04-04cam: Rework how streams configuration is preparedNiklas Söderlund
2019-03-27cam: Separate options valid() and empty()Laurent Pinchart
2019-03-27cam: options: Add support for repeatable optionsNiklas Söderlund
2019-03-27cam: options: Add an array data type to OptionValueNiklas Söderlund
2019-03-25cam: options: Create separate enum for OptionValue typesNiklas Söderlund
2019-03-01cam: Don't requeue requests when stopping streamLaurent Pinchart
2019-02-28libcamera: store stream pointers in sets instead of a vectorsNiklas Söderlund
2019-02-28cam: fix order camera is operated onNiklas Söderlund
PDX-License-Identifier: LGPL-2.1-or-later */ /* * Copyright (C) 2019, Google Inc. * Copyright (C) 2020, Raspberry Pi Ltd * * V4L2 Pixel Format */ #include "libcamera/internal/v4l2_pixelformat.h" #include <ctype.h> #include <map> #include <string.h> #include <libcamera/base/log.h> #include <libcamera/formats.h> #include <libcamera/pixel_format.h> #include "libcamera/internal/formats.h" /** * \file v4l2_pixelformat.h * \brief V4L2 Pixel Format */ namespace libcamera { LOG_DECLARE_CATEGORY(V4L2) /** * \class V4L2PixelFormat * \brief V4L2 pixel format FourCC wrapper * * The V4L2PixelFormat class describes the pixel format of a V4L2 buffer. It * wraps the V4L2 numerical FourCC, and shall be used in all APIs that deal with * V4L2 pixel formats. Its purpose is to prevent unintentional confusion of * V4L2 and DRM FourCCs in code by catching implicit conversion attempts at * compile time. * * To achieve this goal, construction of a V4L2PixelFormat from an integer value * is explicit. To retrieve the integer value of a V4L2PixelFormat, both the * explicit value() and implicit uint32_t conversion operators may be used. */ namespace { const std::map<V4L2PixelFormat, V4L2PixelFormat::Info> vpf2pf{ /* RGB formats. */ { V4L2PixelFormat(V4L2_PIX_FMT_RGB565), { formats::RGB565, "16-bit RGB 5-6-5" } }, { V4L2PixelFormat(V4L2_PIX_FMT_RGB565X), { formats::RGB565_BE, "16-bit RGB 5-6-5 BE" } }, { V4L2PixelFormat(V4L2_PIX_FMT_RGB24), { formats::BGR888, "24-bit RGB 8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_BGR24), { formats::RGB888, "24-bit BGR 8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_XBGR32), { formats::XRGB8888, "32-bit BGRX 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_XRGB32), { formats::BGRX8888, "32-bit XRGB 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_RGBX32), { formats::XBGR8888, "32-bit RGBX 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_BGRX32), { formats::RGBX8888, "32-bit XBGR 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_RGBA32), { formats::ABGR8888, "32-bit RGBA 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_ABGR32), { formats::ARGB8888, "32-bit BGRA 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_ARGB32), { formats::BGRA8888, "32-bit ARGB 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_BGRA32), { formats::RGBA8888, "32-bit ABGR 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_RGB48), { formats::BGR161616, "48-bit RGB 16-16-16" } }, { V4L2PixelFormat(V4L2_PIX_FMT_BGR48), { formats::RGB161616, "48-bit BGR 16-16-16" } }, /* YUV packed formats. */ { V4L2PixelFormat(V4L2_PIX_FMT_YUYV), { formats::YUYV, "YUYV 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YVYU), { formats::YVYU, "YVYU 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_UYVY), { formats::UYVY, "UYVY 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_VYUY), { formats::VYUY, "VYUY 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUVA32), { formats::AVUY8888, "32-bit YUVA 8-8-8-8" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUVX32), { formats::XVUY8888, "32-bit YUVX 8-8-8-8" } }, /* YUV planar formats. */ { V4L2PixelFormat(V4L2_PIX_FMT_NV16), { formats::NV16, "Y/CbCr 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV16M), { formats::NV16, "Y/CbCr 4:2:2 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV61), { formats::NV61, "Y/CrCb 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV61M), { formats::NV61, "Y/CrCb 4:2:2 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV12), { formats::NV12, "Y/CbCr 4:2:0" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV12M), { formats::NV12, "Y/CbCr 4:2:0 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV21), { formats::NV21, "Y/CrCb 4:2:0" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV21M), { formats::NV21, "Y/CrCb 4:2:0 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV24), { formats::NV24, "Y/CbCr 4:4:4" } }, { V4L2PixelFormat(V4L2_PIX_FMT_NV42), { formats::NV42, "Y/CrCb 4:4:4" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUV420), { formats::YUV420, "Planar YUV 4:2:0" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUV420M), { formats::YUV420, "Planar YUV 4:2:0 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YVU420), { formats::YVU420, "Planar YVU 4:2:0" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YVU420M), { formats::YVU420, "Planar YVU 4:2:0 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUV422P), { formats::YUV422, "Planar YUV 4:2:2" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUV422M), { formats::YUV422, "Planar YUV 4:2:2 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YVU422M), { formats::YVU422, "Planar YVU 4:2:2 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YUV444M), { formats::YUV444, "Planar YUV 4:4:4 (N-C)" } }, { V4L2PixelFormat(V4L2_PIX_FMT_YVU444M), { formats::YVU444, "Planar YVU 4:4:4 (N-C)" } }, /* Greyscale formats. */ { V4L2PixelFormat(V4L2_PIX_FMT_GREY), { formats::R8, "8-bit Greyscale" } }, { V4L2PixelFormat(V4L2_PIX_FMT_Y10), { formats::R10, "10-bit Greyscale" } }, { V4L2PixelFormat(V4L2_PIX_FMT_Y10P), { formats::R10_CSI2P, "10-bit Greyscale Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_Y12), { formats::R12, "12-bit Greyscale" } }, { V4L2PixelFormat(V4L2_PIX_FMT_Y12P), { formats::R12_CSI2P, "12-bit Greyscale Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_Y16), { formats::R16, "16-bit Greyscale" } }, /* Bayer formats. */ { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR8), { formats::SBGGR8, "8-bit Bayer BGBG/GRGR" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG8), { formats::SGBRG8, "8-bit Bayer GBGB/RGRG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG8), { formats::SGRBG8, "8-bit Bayer GRGR/BGBG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB8), { formats::SRGGB8, "8-bit Bayer RGRG/GBGB" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR10), { formats::SBGGR10, "10-bit Bayer BGBG/GRGR" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG10), { formats::SGBRG10, "10-bit Bayer GBGB/RGRG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG10), { formats::SGRBG10, "10-bit Bayer GRGR/BGBG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB10), { formats::SRGGB10, "10-bit Bayer RGRG/GBGB" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR10P), { formats::SBGGR10_CSI2P, "10-bit Bayer BGBG/GRGR Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG10P), { formats::SGBRG10_CSI2P, "10-bit Bayer GBGB/RGRG Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG10P), { formats::SGRBG10_CSI2P, "10-bit Bayer GRGR/BGBG Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB10P), { formats::SRGGB10_CSI2P, "10-bit Bayer RGRG/GBGB Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR12), { formats::SBGGR12, "12-bit Bayer BGBG/GRGR" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG12), { formats::SGBRG12, "12-bit Bayer GBGB/RGRG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG12), { formats::SGRBG12, "12-bit Bayer GRGR/BGBG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB12), { formats::SRGGB12, "12-bit Bayer RGRG/GBGB" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR12P), { formats::SBGGR12_CSI2P, "12-bit Bayer BGBG/GRGR Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG12P), { formats::SGBRG12_CSI2P, "12-bit Bayer GBGB/RGRG Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG12P), { formats::SGRBG12_CSI2P, "12-bit Bayer GRGR/BGBG Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB12P), { formats::SRGGB12_CSI2P, "12-bit Bayer RGRG/GBGB Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR14), { formats::SBGGR14, "14-bit Bayer BGBG/GRGR" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG14), { formats::SGBRG14, "14-bit Bayer GBGB/RGRG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG14), { formats::SGRBG14, "14-bit Bayer GRGR/BGBG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB14), { formats::SRGGB14, "14-bit Bayer RGRG/GBGB" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR14P), { formats::SBGGR14_CSI2P, "14-bit Bayer BGBG/GRGR Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG14P), { formats::SGBRG14_CSI2P, "14-bit Bayer GBGB/RGRG Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG14P), { formats::SGRBG14_CSI2P, "14-bit Bayer GRGR/BGBG Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB14P), { formats::SRGGB14_CSI2P, "14-bit Bayer RGRG/GBGB Packed" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SBGGR16), { formats::SBGGR16, "16-bit Bayer BGBG/GRGR" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGBRG16), { formats::SGBRG16, "16-bit Bayer GBGB/RGRG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SGRBG16), { formats::SGRBG16, "16-bit Bayer GRGR/BGBG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_SRGGB16), { formats::SRGGB16, "16-bit Bayer RGRG/GBGB" } }, { V4L2PixelFormat(V4L2_PIX_FMT_PISP_COMP1_BGGR), { formats::BGGR_PISP_COMP1, "16-bit Bayer BGBG/GRGR PiSP Compress Mode 1" } }, { V4L2PixelFormat(V4L2_PIX_FMT_PISP_COMP1_GBRG), { formats::GBRG_PISP_COMP1, "16-bit Bayer GBGB/RGRG PiSP Compress Mode 1" } }, { V4L2PixelFormat(V4L2_PIX_FMT_PISP_COMP1_GRBG), { formats::GRBG_PISP_COMP1, "16-bit Bayer GRGR/BGBG PiSP Compress Mode 1" } }, { V4L2PixelFormat(V4L2_PIX_FMT_PISP_COMP1_RGGB), { formats::RGGB_PISP_COMP1, "16-bit Bayer RGRG/GBGB PiSP Compress Mode 1" } }, { V4L2PixelFormat(V4L2_PIX_FMT_PISP_COMP1_MONO), { formats::MONO_PISP_COMP1, "16-bit Mono PiSP Compress Mode 1" } }, /* Compressed formats. */ { V4L2PixelFormat(V4L2_PIX_FMT_MJPEG), { formats::MJPEG, "Motion-JPEG" } }, { V4L2PixelFormat(V4L2_PIX_FMT_JPEG), { formats::MJPEG, "JPEG JFIF" } }, }; } /* namespace */ /** * \struct V4L2PixelFormat::Info * \brief Information about a V4L2 pixel format * * \var V4L2PixelFormat::Info::format * \brief The corresponding libcamera PixelFormat * * \sa PixelFormat * * \var V4L2PixelFormat::Info::description * \brief The human-readable description of the V4L2 pixel format */ /** * \fn V4L2PixelFormat::V4L2PixelFormat() * \brief Construct a V4L2PixelFormat with an invalid format * * V4L2PixelFormat instances constructed with the default constructor are * invalid, calling the isValid() function returns false. */ /** * \fn V4L2PixelFormat::V4L2PixelFormat(uint32_t fourcc) * \brief Construct a V4L2PixelFormat from a FourCC value * \param[in] fourcc The pixel format FourCC numerical value */ /** * \fn bool V4L2PixelFormat::isValid() const * \brief Check if the pixel format is valid * * V4L2PixelFormat instances constructed with the default constructor are * invalid. Instances constructed with a FourCC defined in the V4L2 API are * valid. The behaviour is undefined otherwise. * * \return True if the pixel format is valid, false otherwise */ /** * \fn uint32_t V4L2PixelFormat::fourcc() const * \brief Retrieve the pixel format FourCC numerical value * \return The pixel format FourCC numerical value */ /** * \fn V4L2PixelFormat::operator uint32_t() const * \brief Convert to the pixel format FourCC numerical value * \return The pixel format FourCC numerical value */ /** * \brief Assemble and return a string describing the pixel format * \return A string describing the pixel format */ std::string V4L2PixelFormat::toString() const { if (fourcc_ == 0) return "<INVALID>"; char ss[8] = { static_cast<char>(fourcc_ & 0x7f), static_cast<char>((fourcc_ >> 8) & 0x7f), static_cast<char>((fourcc_ >> 16) & 0x7f), static_cast<char>((fourcc_ >> 24) & 0x7f) }; for (unsigned int i = 0; i < 4; i++) { if (!isprint(ss[i])) ss[i] = '.'; } if (fourcc_ & (1 << 31)) strcat(ss, "-BE"); return ss; } /** * \brief Retrieve the V4L2 description for the format * * The description matches the value used by the kernel, as would be reported * by the VIDIOC_ENUM_FMT ioctl. * * \return The V4L2 description corresponding to the V4L2 format, or a * placeholder description if not found */ const char *V4L2PixelFormat::description() const { const auto iter = vpf2pf.find(*this); if (iter == vpf2pf.end()) { LOG(V4L2, Warning) << "Unsupported V4L2 pixel format " << toString(); return "Unsupported format"; } return iter->second.description; } /** * \brief Convert the V4L2 pixel format to the corresponding PixelFormat * \param[in] warn When true, log a warning message if the V4L2 pixel format * isn't known * * Users of this function might try to convert a V4L2PixelFormat to a * PixelFormat just to check if the format is supported or not. In that case, * they can suppress the warning message by setting the \a warn argument to * false to not pollute the log with unnecessary messages. * * \return The PixelFormat corresponding to the V4L2 pixel format */ PixelFormat V4L2PixelFormat::toPixelFormat(bool warn) const { const auto iter = vpf2pf.find(*this); if (iter == vpf2pf.end()) { if (warn) LOG(V4L2, Warning) << "Unsupported V4L2 pixel format " << toString(); return PixelFormat(); } return iter->second.format; } /** * \brief Retrieve the list of V4L2PixelFormat associated with \a pixelFormat * \param[in] pixelFormat The PixelFormat to convert * * Multiple V4L2 formats may exist for one PixelFormat as V4L2 defines separate * 4CCs for contiguous and non-contiguous versions of the same image format. * * \return The list of V4L2PixelFormat corresponding to \a pixelFormat */ const std::vector<V4L2PixelFormat> & V4L2PixelFormat::fromPixelFormat(const PixelFormat &pixelFormat) { static const std::vector<V4L2PixelFormat> empty; const PixelFormatInfo &info = PixelFormatInfo::info(pixelFormat); if (!info.isValid()) return empty; return info.v4l2Formats; } /** * \brief Insert a text representation of a V4L2PixelFormat into an output * stream * \param[in] out The output stream * \param[in] f The V4L2PixelFormat * \return The output stream \a out */ std::ostream &operator<<(std::ostream &out, const V4L2PixelFormat &f) { out << f.toString(); return out; } } /* namespace libcamera */