bindings: cxx: add line_config.set_output_values()
authorBartosz Golaszewski <bartosz.golaszewski@linaro.org>
Thu, 12 Jan 2023 18:30:25 +0000 (19:30 +0100)
committerBartosz Golaszewski <bartosz.golaszewski@linaro.org>
Wed, 25 Jan 2023 08:42:02 +0000 (09:42 +0100)
Extend line_config to expose a new method - set_output_values() - which
wraps the new C function for setting multiple output values at once.

Signed-off-by: Bartosz Golaszewski <bartosz.golaszewski@linaro.org>
bindings/cxx/gpiodcxx/line-config.hpp
bindings/cxx/internal.hpp
bindings/cxx/line-config.cpp
bindings/cxx/line-settings.cpp
bindings/cxx/tests/tests-line-config.cpp

index a91791338db9f0578e02ca37a914c4445e8bace3..b76fdff4a5a0aa6c9f12d640686838e196346b7a 100644 (file)
@@ -76,6 +76,13 @@ public:
         */
        line_config& add_line_settings(const line::offsets& offsets, const line_settings& settings);
 
+       /**
+        * @brief Set output values for a number of lines.
+        * @param values Buffer containing the output values.
+        * @return Reference to self.
+        */
+       line_config& set_output_values(const line::values& values);
+
        /**
         * @brief Get a mapping of offsets to line settings stored by this
         *        object.
index 6aceac19031c7b5bc8afbbbacf89721af2a3f35e..8322e1262c36114944d703e69c6af5484c78e921 100644 (file)
@@ -31,6 +31,7 @@ map_enum_c_to_cxx(c_enum_type value, const ::std::map<c_enum_type, cxx_enum_type
 }
 
 void throw_from_errno(const ::std::string& what);
+::gpiod_line_value map_output_value(line::value value);
 
 template<class T, void F(T*)> struct deleter
 {
index 3ec99f0b0214ef40bf5ddcd292f8937d2dfd26ad..233ba33ba2a6b97cc9db9ecb09bc20f902e377f0 100644 (file)
@@ -100,6 +100,21 @@ GPIOD_CXX_API line_config& line_config::add_line_settings(const line::offsets& o
        return *this;
 }
 
+GPIOD_CXX_API line_config& line_config::set_output_values(const line::values& values)
+{
+       ::std::vector<::gpiod_line_value> mapped_values(values.size());
+
+       for (unsigned int i = 0; i < values.size(); i++)
+               mapped_values[i] = map_output_value(values[i]);
+
+       auto ret = ::gpiod_line_config_set_output_values(this->_m_priv->config.get(),
+                                                        mapped_values.data(), mapped_values.size());
+       if (ret)
+               throw_from_errno("unable to set output values");
+
+       return *this;
+}
+
 GPIOD_CXX_API ::std::map<line::offset, line_settings> line_config::get_line_settings() const
 {
        ::std::size_t num_offsets = ::gpiod_line_config_get_num_configured_offsets(
index 32f21a35f73b29cb85bbd9129f4898c402da855d..21590621b9c12076ed0a8c244a1d04ea500aecef 100644 (file)
@@ -139,6 +139,11 @@ cxx_enum_type get_mapped_value(::gpiod_line_settings* settings,
 
 } /* namespace */
 
+::gpiod_line_value map_output_value(line::value value)
+{
+       return do_map_value(value, value_mapping);
+}
+
 line_settings::impl::impl()
        : settings(make_line_settings())
 {
index 5fa0f94232742f7b65b5f51222de5d12f8db76af..5e439a10336673ef5c838b80708f7ddeedeabfda 100644 (file)
@@ -4,12 +4,17 @@
 #include <catch2/catch.hpp>
 #include <gpiod.hpp>
 
+#include "gpiosim.hpp"
 #include "helpers.hpp"
 
+using ::gpiosim::make_sim;
 using namespace ::std::chrono_literals;
 using direction = ::gpiod::line::direction;
 using drive = ::gpiod::line::drive;
 using edge = ::gpiod::line::edge;
+using simval = ::gpiosim::chip::value;
+using value = ::gpiod::line::value;
+using values = ::gpiod::line::values;
 
 namespace {
 
@@ -72,6 +77,52 @@ TEST_CASE("line_config can be reset", "[line-config]")
        REQUIRE(cfg.get_line_settings().size() == 0);
 }
 
+TEST_CASE("output values can be set globally", "[line-config]")
+{
+       const values vals = { value::ACTIVE, value::INACTIVE, value::ACTIVE, value::INACTIVE };
+
+       auto sim = make_sim()
+               .set_num_lines(4)
+               .build();
+
+       ::gpiod::line_config cfg;
+
+       SECTION("request with globally set output values")
+       {
+               cfg
+                       .add_line_settings(
+                               {0, 1, 2, 3},
+                               ::gpiod::line_settings().set_direction(direction::OUTPUT)
+                       )
+                       .set_output_values(vals);
+
+               auto request = ::gpiod::chip(sim.dev_path())
+                       .prepare_request()
+                       .set_line_config(cfg)
+                       .do_request();
+
+               REQUIRE(sim.get_value(0) == simval::ACTIVE);
+               REQUIRE(sim.get_value(1) == simval::INACTIVE);
+               REQUIRE(sim.get_value(2) == simval::ACTIVE);
+               REQUIRE(sim.get_value(3) == simval::INACTIVE);
+       }
+
+       SECTION("read back global output values")
+       {
+               cfg
+                       .add_line_settings(
+                               {0, 1, 2, 3},
+                               ::gpiod::line_settings()
+                                       .set_direction(direction::OUTPUT)
+                                       .set_output_value(value::ACTIVE)
+                       )
+                       .set_output_values(vals);
+
+               auto settings = cfg.get_line_settings()[1];
+               REQUIRE(settings.output_value() == value::INACTIVE);
+       }
+}
+
 TEST_CASE("line_config stream insertion operator works", "[line-config]")
 {
        ::gpiod::line_config cfg;