From patchwork Mon Dec 2 13:34:00 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Jacopo Mondi X-Patchwork-Id: 22147 Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id 1C7BFBEFBE for ; Mon, 2 Dec 2024 13:34:17 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id 8C53766069; Mon, 2 Dec 2024 14:34:13 +0100 (CET) Authentication-Results: lancelot.ideasonboard.com; dkim=pass (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="NWJR6iRC"; dkim-atps=neutral Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [IPv6:2001:4b98:dc2:55:216:3eff:fef7:d647]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id 929AD66040 for ; Mon, 2 Dec 2024 14:34:10 +0100 (CET) Received: from ideasonboard.com (mob-5-90-236-68.net.vodafone.it [5.90.236.68]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id A64D275A; Mon, 2 Dec 2024 14:33:43 +0100 (CET) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1733146424; bh=4clINLyOxNeTfw/ssU7qARDlzT04DLr1FwFQfBDRrVY=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=NWJR6iRCP+hOVExUglJ+foKJlMY0rkXJDqEzqsUm+dyC4b8oLb3GU6UQBlcld+LM0 mSyh8CR/kK72Ru+2mM7DLVNyjd5XAXl/vhV366aR0+YicrFCeUvV9GqqWpwfBqbFQb RhFNgGsAHLngvluCx6hdTphByp/3MU/wFNnDlv54= From: Jacopo Mondi To: libcamera-devel@lists.libcamera.org Cc: Jacopo Mondi Subject: [PATCH v2 1/4] libcamera: Implement YamlEmitter Date: Mon, 2 Dec 2024 14:34:00 +0100 Message-ID: <20241202133404.41431-2-jacopo.mondi@ideasonboard.com> X-Mailer: git-send-email 2.47.0 In-Reply-To: <20241202133404.41431-1-jacopo.mondi@ideasonboard.com> References: <20241202133404.41431-1-jacopo.mondi@ideasonboard.com> MIME-Version: 1.0 X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" Implement a helpers to allow outputting text in YAML format. The class of helpers allows to create list and dictionaries and emit scalar values starting from a YamlRoot object initialized with a file path. Signed-off-by: Jacopo Mondi --- include/libcamera/internal/meson.build | 1 + include/libcamera/internal/yaml_emitter.h | 149 +++++ src/libcamera/meson.build | 1 + src/libcamera/yaml_emitter.cpp | 719 ++++++++++++++++++++++ 4 files changed, 870 insertions(+) create mode 100644 include/libcamera/internal/yaml_emitter.h create mode 100644 src/libcamera/yaml_emitter.cpp diff --git a/include/libcamera/internal/meson.build b/include/libcamera/internal/meson.build index 1dddcd50c90b..357d96566cfb 100644 --- a/include/libcamera/internal/meson.build +++ b/include/libcamera/internal/meson.build @@ -42,6 +42,7 @@ libcamera_internal_headers = files([ 'v4l2_pixelformat.h', 'v4l2_subdevice.h', 'v4l2_videodevice.h', + 'yaml_emitter.h', 'yaml_parser.h', ]) diff --git a/include/libcamera/internal/yaml_emitter.h b/include/libcamera/internal/yaml_emitter.h new file mode 100644 index 000000000000..22b3c43b919d --- /dev/null +++ b/include/libcamera/internal/yaml_emitter.h @@ -0,0 +1,149 @@ +/* SPDX-License-Identifier: LGPL-2.1-or-later */ +/* + * Copyright (C) 2024, Ideas On Board Oy + * + * libcamera YAML emitter helper + */ + +#pragma once + +#include +#include +#include + +#include +#include + +#include + +namespace libcamera { + +class YamlDict; +class YamlEvent; +class YamlList; +class YamlRoot; +class YamlScalar; + +class YamlEmitter final +{ +public: + ~YamlEmitter(); + + static YamlRoot root(const std::string &path); + +private: + friend class YamlOutput; + friend class YamlRoot; + + LIBCAMERA_DISABLE_COPY(YamlEmitter) + + YamlEmitter(const std::string &path); + + void logError(); + void init(); + int emit(); + + File file_; + yaml_event_t event_; + yaml_emitter_t emitter_; +}; + +class YamlOutput +{ +public: + bool valid() const { return !!emitter_; } + +protected: + YamlOutput() = default; + YamlOutput(YamlOutput &&other); + YamlOutput(YamlEmitter *emitter, YamlOutput *parent); + + virtual ~YamlOutput(); + + YamlOutput &operator=(YamlOutput &&other); + + int emitScalar(std::string_view scalar); + int emitMappingStart(); + int emitMappingEnd(); + int emitSequenceStart(); + int emitSequenceEnd(); + + YamlDict dict(YamlOutput *parent); + YamlList list(YamlOutput *parent); + + YamlEmitter *emitter_ = nullptr; + yaml_event_t event_; + + YamlOutput *parent_ = nullptr; + YamlOutput *child_ = nullptr; + +private: + LIBCAMERA_DISABLE_COPY(YamlOutput) + + void unlinkChildren(YamlOutput *node); +}; + +class YamlRoot : public YamlOutput +{ +public: + YamlRoot() = default; + YamlRoot(YamlRoot &&other) = default; + ~YamlRoot(); + + YamlRoot &operator=(YamlRoot &&other) = default; + + YamlList list(); + YamlDict dict(); + +private: + LIBCAMERA_DISABLE_COPY(YamlRoot) + + friend class YamlEmitter; + + YamlRoot(std::unique_ptr emitter) + : YamlOutput(emitter.get(), nullptr), emitterRoot_(std::move(emitter)) + { + } + + std::unique_ptr emitterRoot_; +}; + +class YamlList : public YamlOutput +{ +public: + YamlList() = default; + YamlList(YamlList &&other) = default; + ~YamlList(); + + YamlList &operator=(YamlList &&other) = default; + + YamlList list(); + YamlDict dict(); + void scalar(std::string_view scalar); + +private: + friend class YamlOutput; + + YamlList(YamlEmitter *emitter, YamlOutput *parent); +}; + +class YamlDict : public YamlOutput +{ +public: + YamlDict() = default; + YamlDict(YamlDict &&other) = default; + ~YamlDict(); + + YamlDict &operator=(YamlDict &&other) = default; + + YamlList list(std::string_view key); + YamlDict dict(std::string_view key); + void scalar(std::string_view key, std::string_view scalar); + +private: + friend class YamlOutput; + + YamlDict(YamlEmitter *emitter, YamlOutput *parent); +}; + +} /* namespace libcamera */ diff --git a/src/libcamera/meson.build b/src/libcamera/meson.build index 21cae11756d6..933b98bf05e9 100644 --- a/src/libcamera/meson.build +++ b/src/libcamera/meson.build @@ -52,6 +52,7 @@ libcamera_internal_sources = files([ 'v4l2_pixelformat.cpp', 'v4l2_subdevice.cpp', 'v4l2_videodevice.cpp', + 'yaml_emitter.cpp', 'yaml_parser.cpp', ]) diff --git a/src/libcamera/yaml_emitter.cpp b/src/libcamera/yaml_emitter.cpp new file mode 100644 index 000000000000..c0629e249c57 --- /dev/null +++ b/src/libcamera/yaml_emitter.cpp @@ -0,0 +1,719 @@ +/* SPDX-License-Identifier: LGPL-2.1-or-later */ +/* + * Copyright (C) 2024, Ideas On Board Oy + * + * libcamera YAML emitter helper + */ + +#include "libcamera/internal/yaml_emitter.h" + +#include + +/** + * \file yaml_emitter.h + * \brief A YAML emitter helper + * + * The YAML emitter helpers allow users to emit output in YAML format. + + * Unlike YAML writers that operate on a fully populated representation of the + * data, the YAML emitter outputs YAML data on the fly. It is suitable for + * outputting large amount of data with low overhead and no runtime heap + * allocation. + * + * To emit YAML users of the these helper classes create a root node with + * + * \code + std::string filePath("..."); + auto root = YamlEmitter::root(filePath); + \endcode + * + * and start emitting dictionaries and lists with the YamlRoot::dict() and + * YamlRoot::list() functions. + * + * The classes part of this file implement RAII-style handling of YAML + * events. By creating a YamlList and YamlDict instance the associated YAML + * sequence start and mapping start events are emitted and once the instances + * gets destroyed the corresponding sequence end and mapping end events are + * emitted. + * + * From an initialized YamlRoot instance is possible to create YAML list and + * dictionaries. + * + * \code + YamlDict dict = root.dict(); + YamlList list = root.list(); + \endcode + * + * YamlDict instances can be populated with scalars associated with a key + * + * \code + dict["key"] = "value"; + \endcode + * + * and it is possible to create lists and dictionaries, associated with a key + * + * \code + YamlDict subDict = dict.dict("newDict"); + YamlList subList = dict.list("newList"); + \endcode + * + * YamlList instances can be populated with scalar elements + * + * \code + list.scalar("x"); + list.scalar("y"); + \endcode + * + * and with dictionaries and lists too + * + * \code + YamlDict subDict = list.dict(); + YamlList subList = list.list(); + \endcode + */ + +namespace libcamera { + +LOG_DEFINE_CATEGORY(YamlEmitter) + +namespace { + +int yamlWrite(void *data, unsigned char *buffer, size_t size) +{ + File *file = static_cast(data); + + Span buf{ buffer, size }; + ssize_t ret = file->write(buf); + if (ret < 0) { + LOG(YamlEmitter, Error) << "Write error: " << strerror(ret); + return 0; + } + + return 1; +} + +} /* namespace */ + +/** + * \class YamlEmitter + * + * YAML helper classes entry point. This class allows to create a YamlRoot + * instance, using the YamlEmitter::root() function, that users can populate + * with lists, dictionaries and scalars. + */ + +YamlEmitter::YamlEmitter(const std::string &path) +{ + file_.setFileName(path); + file_.open(File::OpenModeFlag::WriteOnly); +} + +YamlEmitter::~YamlEmitter() +{ + yaml_event_delete(&event_); + yaml_emitter_delete(&emitter_); +} + +/** + * \brief Create an initialized instance of YamlRoot + * \param[in] path The YAML output file path + * + * Create an initialized instance of the YamlRoot class that users can start + * using and populating with scalars, lists and dictionaries. + * + * \return An initialized YamlRoot instance + */ +YamlRoot YamlEmitter::root(const std::string &path) +{ + std::unique_ptr emitter{ new YamlEmitter(path) }; + + emitter->init(); + + return YamlRoot(std::move(emitter)); +} + +void YamlEmitter::logError() +{ + switch (emitter_.error) { + case YAML_MEMORY_ERROR: + LOG(YamlEmitter, Error) + << "Memory error: Not enough memory for emitting"; + break; + + case YAML_WRITER_ERROR: + LOG(YamlEmitter, Error) + << "Writer error: " << emitter_.problem; + break; + + case YAML_EMITTER_ERROR: + LOG(YamlEmitter, Error) + << "Emitter error: " << emitter_.problem; + break; + + default: + LOG(YamlEmitter, Error) << "Internal error"; + break; + } +} + +void YamlEmitter::init() +{ + yaml_emitter_initialize(&emitter_); + yaml_emitter_set_output(&emitter_, yamlWrite, &file_); + + yaml_stream_start_event_initialize(&event_, YAML_UTF8_ENCODING); + emit(); + + yaml_document_start_event_initialize(&event_, nullptr, nullptr, + nullptr, 0); + emit(); +} + +int YamlEmitter::emit() +{ + int ret = yaml_emitter_emit(&emitter_, &event_); + if (!ret) { + logError(); + return -EINVAL; + } + + return 0; +} + +/** + * \class YamlOutput + * + * The YamlOutput base class. From this class are derived the YamlList and + * YamlDict classes which are meant to be used by users of the YAML emitter + * helpers. + * + * The YamlOutput base class provides functions to create YAML lists and + * dictionaries and to populate them. + * + * This class cannot be instantiated directly. + */ + +/** + * \fn YamlOutput::YamlOutput() + * \brief The default constructor + * + * Create an empty, non-initialized instance. The newly created instance cannot + * be used without being first move-assigned to a valid and initialized + * instance. Valid instances of this class can only be created using the + * YamlRoot::list(), YamlRoot::dict(), YamlList::list(), YamlList::dict(), + * YamlDict::list() and YamlDict::dict() functions. + */ + +/** + * \brief Create a YamlOutput instance with an associated \a emitter and \a parent + * \param[in] emitter The YAML emitter + * \param[in] parent Pointer to the parent node + * + * Create a YamlOutput with a valid YamlEmitter and with a reference to the + * parent node. The parent's child_ pointer is assigned to the newly created + * instance. If the parent already has a valid child_ reference, unlink and + * invalidate the chain of previously assigned children nodes by resetting their + * parent_ pointer. None of the children nodes can be used anymore from this + * point on. + */ +YamlOutput::YamlOutput(YamlEmitter *emitter, YamlOutput *parent) + : emitter_(emitter), parent_(parent) +{ + if (!parent_) + return; + + unlinkChildren(parent_->child_); + parent_->child_ = this; +} + +/** + * \brief Move constructor + * \param[in] other The instance to be moved + * + * Move an instance and invalidate \a other. The move constructor reassigns + * the parent's child_ to point to the newly initialized instance. + */ +YamlOutput::YamlOutput(YamlOutput &&other) +{ + emitter_ = other.emitter_; + event_ = other.event_; + parent_ = other.parent_; + child_ = other.child_; + + other.emitter_ = nullptr; + other.event_ = {}; + other.parent_ = nullptr; + other.child_ = nullptr; + + if (parent_) + parent_->child_ = this; +} + +YamlOutput::~YamlOutput() +{ + if (parent_) + parent_->child_ = nullptr; +} + +void YamlOutput::unlinkChildren(YamlOutput *node) +{ + YamlOutput *nextNode = node; + while (nextNode) { + YamlOutput *nextChild = nextNode->child_; + + nextNode->parent_ = nullptr; + nextNode->child_ = nullptr; + nextNode = nextChild; + } +} + +/** + * \brief Move assignment operator + * \param[in] other The instance to be moved + * + * Move-assign an instance and invalidate \a other. The move assignment operator + * reassigns the parent's child_ to point to the newly initialized instance. + */ +YamlOutput &YamlOutput::operator=(YamlOutput &&other) +{ + emitter_ = other.emitter_; + event_ = other.event_; + parent_ = other.parent_; + child_ = other.child_; + + other.emitter_ = nullptr; + other.event_ = {}; + other.parent_ = nullptr; + other.child_ = nullptr; + + if (parent_) + parent_->child_ = this; + + return *this; +} + +/** + * \fn YamlOutput::valid() + * \brief Check if a YamlOutput instance has been correctly initialized + * \return True if the instance has been initialized, false otherwise + */ + +/** + * \brief Emit \a scalar as a YAML scalar + * \param[in] scalar The element to emit + * \return 0 in case of success, a negative error value otherwise + */ +int YamlOutput::emitScalar(std::string_view scalar) +{ + if (!valid()) + return -EINVAL; + + const yaml_char_t *value = reinterpret_cast + (scalar.data()); + + /* + * \todo Remove the const_cast of 'value' when + * dropping support for libyaml versions < 0.2.3 (shipped in Debian11). + * + * https://github.com/yaml/libyaml/pull/140 + */ + yaml_scalar_event_initialize(&emitter_->event_, nullptr, nullptr, + const_cast(value), + scalar.length(), true, false, + YAML_PLAIN_SCALAR_STYLE); + return emitter_->emit(); +} + +/** + * \brief Emit the mapping start YAML event + * \return 0 in case of success, a negative error value otherwise + */ +int YamlOutput::emitMappingStart() +{ + if (!valid()) + return -EINVAL; + + yaml_mapping_start_event_initialize(&emitter_->event_, nullptr, nullptr, + true, YAML_BLOCK_MAPPING_STYLE); + return emitter_->emit(); +} + +/** + * \brief Emit the mapping end YAML event + * \return 0 in case of success, a negative error value otherwise + */ +int YamlOutput::emitMappingEnd() +{ + if (!valid()) + return -EINVAL; + + yaml_mapping_end_event_initialize(&emitter_->event_); + return emitter_->emit(); +} + +/** + * \brief Emit the sequence start YAML event + * \return 0 in case of success, a negative error value otherwise + */ +int YamlOutput::emitSequenceStart() +{ + if (!valid()) + return -EINVAL; + + yaml_sequence_start_event_initialize(&emitter_->event_, nullptr, nullptr, + true, YAML_BLOCK_SEQUENCE_STYLE); + return emitter_->emit(); +} + +/** + * \brief Emit the sequence end YAML event + * \return 0 in case of success, a negative error value otherwise + */ +int YamlOutput::emitSequenceEnd() +{ + if (!valid()) + return -EINVAL; + + yaml_sequence_end_event_initialize(&emitter_->event_); + return emitter_->emit(); +} + +/** + * \brief Create a dictionary instance + * \return An instance of YamlDict + */ +YamlDict YamlOutput::dict(YamlOutput *parent) +{ + return YamlDict(emitter_, parent); +} + +/** + * \brief Create a list instance + * \return An instance of YamlList + */ +YamlList YamlOutput::list(YamlOutput *parent) +{ + return YamlList(emitter_, parent); +} + +/** + * \var YamlOutput::emitter_ + * \brief The emitter used by this YamlObject to output YAML events + */ + +/** + * \var YamlOutput::event_ + * \brief The YAML event used by this YamlObject + */ + +/** + * \var YamlOutput::parent_ + * \brief The parent node, set at object creation time + * + * The parent_ class member references the parent node. Only nodes with a valid + * parent can emit valid YAML output. The parent_ pointer is reset to an invalid + * value when a new child is created on the parent node, effectively + * invalidating the current instance. + */ + +/** + * \var YamlOutput::child_ + * \brief The child node + * + * The child_ class member references any eventual child created from the + * current instance using the list() and dict() functions. Only a single + * valid child_ at the time can be assigned to a YamlOutput instance, if + * multiple child are created the last created ones overwrites the child_ + * pointer and invalidates all other children. + */ + +/** + * \class YamlRoot + * + * The YAML root node. A valid YamlRoot instance can only be created using the + * YamlEmitter::root() function. The typical initialization pattern of users of + * this class is similar to the one in the following example: + * + * \code + class YamlUser + { + public: + YamlUser(); + + private: + YamlRool root_; + }; + + YamlUser::YamlUser() + { + root_ = YamlEmitter::root("/path/to/yaml/file.yml"); + } + \endcode + * + * A YamlRoot element can be populated with list and dictionaries. + */ + +/** + * \fn YamlRoot::YamlRoot() + * \brief Construct a YamlRoot instance without initializing it + * + * A YamlRoot instance can be created in non-initialized state typically to be + * stored as a class member to maintain their lifetime active. In order to start + * using and populating a YamlRoot, a valid and initialized instance created + * using the YamlEmitter::root() function has to be move-assigned to a + * non-initialized instance. + * + * \code + YamlRoot root; + + root = YamlEmitter::root("/path/to/yaml/file.yml"); + \endcode + */ + +/** + * \fn YamlRoot::YamlRoot() + * \copydoc YamlOutput::YamlOutput() + */ + +/** + * \fn YamlRoot &YamlRoot::YamlRoot(YamlRoot &&other) + * \copydoc YamlOutput &YamlOutput::YamlOutput(YamlOutput &&other) + */ + +YamlRoot::~YamlRoot() +{ + if (!valid()) + return; + + yaml_document_end_event_initialize(&emitter_->event_, 0); + emitterRoot_->emit(); + + yaml_stream_end_event_initialize(&emitter_->event_); + emitterRoot_->emit(); +} + +/** + * \fn YamlRoot &YamlRoot::operator=(YamlRoot &&other) + * \copydoc YamlOutput &operator=(YamlOutput &&other) + */ + +/** + * \copydoc YamlOutput::dict() + */ +YamlDict YamlRoot::dict() +{ + int ret = emitMappingStart(); + if (ret) + return {}; + + return YamlOutput::dict(this); +} + +/** + * \copydoc YamlOutput::list() + */ +YamlList YamlRoot::list() +{ + int ret = emitSequenceStart(); + if (ret) + return {}; + + return YamlOutput::list(this); +} + +/** + * \class YamlList + * + * A YamlList can be populated with scalars and allows to create nested lists + * and dictionaries. + */ + +/** + * \fn YamlList::YamlList() + * \copydoc YamlOutput::YamlOutput() + */ + +/** + * \copydoc YamlOutput::YamlOutput(YamlEmitter *emitter, YamlOutput *parent) + */ +YamlList::YamlList(YamlEmitter *emitter, YamlOutput *parent) + : YamlOutput(emitter, parent) +{ +} + +/** + * \fn YamlList &YamlList::YamlList(YamlList &&other) + * \copydoc YamlOutput &YamlOutput::YamlOutput(YamlOutput &&other) + */ + +YamlList::~YamlList() +{ + emitSequenceEnd(); +} + +/** + * \fn YamlList &YamlList::operator=(YamlList &&other) + * \copydoc YamlOutput &operator=(YamlOutput &&other) + */ + +/** + * \copydoc YamlOutput::list() + */ +YamlList YamlList::list() +{ + if (!parent_) { + LOG(YamlEmitter, Error) + << "Invalid usage of the YamlEmitter API. " + << " The YAML output might not be correct."; + return {}; + } + + int ret = emitSequenceStart(); + if (ret) + return {}; + + return YamlOutput::list(this); +} + +/** + * \copydoc YamlOutput::dict() + */ +YamlDict YamlList::dict() +{ + if (!parent_) { + LOG(YamlEmitter, Error) + << "Invalid usage of the YamlEmitter API. " + << " The YAML output might not be correct."; + return {}; + } + + int ret = emitMappingStart(); + if (ret) + return {}; + + return YamlOutput::dict(this); +} + +/** + * \brief Append \a scalar to the list + * \param[in] scalar The element to append to the list + */ +void YamlList::scalar(std::string_view scalar) +{ + if (!parent_) { + LOG(YamlEmitter, Error) + << "Invalid usage of the YamlEmitter API. " + << " The YAML output might not be correct."; + return; + } + + emitScalar(scalar); +} + +/** + * \class YamlDict + * + * A YamlDict can create lists, other dictionaries and emit a scalar associated + * with a key. + */ + +/** + * \fn YamlDict::YamlDict() + * \copydoc YamlOutput::YamlOutput() + */ + +/** + * \copydoc YamlOutput::YamlOutput(YamlEmitter *emitter, YamlOutput *parent) + */ +YamlDict::YamlDict(YamlEmitter *emitter, YamlOutput *parent) + : YamlOutput(emitter, parent) +{ +} + +/** + * \fn YamlDict &YamlDict::YamlDict(YamlDict &&other) + * \copydoc YamlOutput &YamlOutput::YamlOutput(YamlOutput &&other) + */ + +YamlDict::~YamlDict() +{ + emitMappingEnd(); +} + +/** + * \fn YamlDict &YamlDict::operator=(YamlDict &&other) + * \copydoc YamlOutput &operator=(YamlOutput &&other) + */ + +/** + * \brief Create a list associated with \a key + * \param[in] key The key to associate the list with + * \return An instance of YamlList + */ +YamlList YamlDict::list(std::string_view key) +{ + if (!parent_) { + LOG(YamlEmitter, Error) + << "Invalid usage of the YamlEmitter API. " + << " The YAML output might not be correct."; + return {}; + } + + int ret = emitScalar(key); + if (ret) + return {}; + + ret = emitSequenceStart(); + if (ret) + return {}; + + return YamlOutput::list(this); +} + +/** + * \brief Create a dictionary associated with \a key + * \param[in] key The key to associate the dictionary with + * \return An instance of YamlDict + */ +YamlDict YamlDict::dict(std::string_view key) +{ + if (!parent_) { + LOG(YamlEmitter, Error) + << "Invalid usage of the YamlEmitter API. " + << " The YAML output might not be correct."; + return {}; + } + + int ret = emitScalar(key); + if (ret) + return {}; + + ret = emitMappingStart(); + if (ret) + return {}; + + return YamlOutput::dict(this); +} + +/** + * \brief Emit \a scalar associated with \a key in the dictionary + * \param[in] key The key associated with the newly created scalar + * \param[in] scalar The scalar to emit + */ +void YamlDict::scalar(std::string_view key, std::string_view scalar) +{ + if (!parent_) { + LOG(YamlEmitter, Error) + << "Invalid usage of the YamlEmitter API. " + << " The YAML output might not be correct."; + return; + } + + int ret = emitScalar(key); + if (ret) + return; + + emitScalar(scalar); +} + +} /* namespace libcamera */