From a951db18fd0edfe19e9e75b91bb30dfa731dbd9c Mon Sep 17 00:00:00 2001 From: Josh Blum Date: Sat, 25 Jun 2011 11:00:47 -0700 Subject: uhd: work on templated property class --- host/include/uhd/CMakeLists.txt | 1 + host/include/uhd/property.hpp | 81 +++++++++++++++++++++++++++++++++++++++++ host/tests/CMakeLists.txt | 1 + host/tests/property_test.cpp | 75 ++++++++++++++++++++++++++++++++++++++ 4 files changed, 158 insertions(+) create mode 100644 host/include/uhd/property.hpp create mode 100644 host/tests/property_test.cpp diff --git a/host/include/uhd/CMakeLists.txt b/host/include/uhd/CMakeLists.txt index 74dc4a7d6..c3c51279c 100644 --- a/host/include/uhd/CMakeLists.txt +++ b/host/include/uhd/CMakeLists.txt @@ -26,6 +26,7 @@ INSTALL(FILES convert.hpp device.hpp exception.hpp + property.hpp version.hpp wax.hpp DESTINATION ${INCLUDE_DIR}/uhd diff --git a/host/include/uhd/property.hpp b/host/include/uhd/property.hpp new file mode 100644 index 000000000..5b47f9482 --- /dev/null +++ b/host/include/uhd/property.hpp @@ -0,0 +1,81 @@ +// +// Copyright 2011 Ettus Research LLC +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program. If not, see . +// + +#ifndef INCLUDED_UHD_PROPERTY_HPP +#define INCLUDED_UHD_PROPERTY_HPP + +#include +#include +#include +#include + +namespace uhd{ + +template class property{ +public: + typedef boost::function subscriber_type; + typedef boost::function master_type; + + /*! + * Register a master subscriber into the property. + * A master is a special subscriber that coerces the value. + * Only one master may be registered per property. + * Registering a master replaces the previous master. + */ + void subscribe_master(const master_type &master){ + _master = master; + } + + /*! + * Register a subscriber into the property. + * All subscribers are called when the value changes. + * Once a subscriber is registered, it cannot be unregistered. + */ + void subscribe(const subscriber_type &subscriber){ + _subscribers.push_back(subscriber); + } + + //! Update calls all subscribers w/ the current value + void update(void){ + this->set(this->get()); + } + + /*! + * Set the new value and call all subscribers. + * The master is called first to coerce the value. + */ + void set(const T &value){ + _value = _master.empty()? value : _master(value); + BOOST_FOREACH(subscriber_type &subscriber, _subscribers){ + subscriber(_value); //let errors propagate + } + } + + //! Get the current value of this property + T get(void) const{ + return _value; + } + +private: + std::list _subscribers; + master_type _master; + T _value; +}; + +} //namespace uhd + +#endif /* INCLUDED_UHD_PROPERTY_HPP */ diff --git a/host/tests/CMakeLists.txt b/host/tests/CMakeLists.txt index b7bcfb7d5..133763591 100644 --- a/host/tests/CMakeLists.txt +++ b/host/tests/CMakeLists.txt @@ -27,6 +27,7 @@ SET(test_sources error_test.cpp gain_group_test.cpp msg_test.cpp + property_test.cpp ranges_test.cpp sph_recv_test.cpp sph_send_test.cpp diff --git a/host/tests/property_test.cpp b/host/tests/property_test.cpp new file mode 100644 index 000000000..c5ef7f1c9 --- /dev/null +++ b/host/tests/property_test.cpp @@ -0,0 +1,75 @@ +// +// Copyright 2011 Ettus Research LLC +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program. If not, see . +// + +#include +#include +#include + +struct coercer_type{ + int doit(int x){ + return x & ~0x3; + } +}; + +struct setter_type{ + void doit(int x){ + _x = x; + } + + int _x; +}; + +BOOST_AUTO_TEST_CASE(test_prop_simple){ + uhd::property prop; + prop.set(42); + BOOST_CHECK_EQUAL(prop.get(), 42); + prop.set(34); + BOOST_CHECK_EQUAL(prop.get(), 34); +} + +BOOST_AUTO_TEST_CASE(test_prop_with_subscriber){ + uhd::property prop; + + setter_type setter; + prop.subscribe(boost::bind(&setter_type::doit, &setter, _1)); + + prop.set(42); + BOOST_CHECK_EQUAL(prop.get(), 42); + BOOST_CHECK_EQUAL(setter._x, 42); + + prop.set(34); + BOOST_CHECK_EQUAL(prop.get(), 34); + BOOST_CHECK_EQUAL(setter._x, 34); +} + +BOOST_AUTO_TEST_CASE(test_prop_with_coercion){ + uhd::property prop; + + setter_type setter; + prop.subscribe(boost::bind(&setter_type::doit, &setter, _1)); + + coercer_type coercer; + prop.subscribe_master(boost::bind(&coercer_type::doit, &coercer, _1)); + + prop.set(42); + BOOST_CHECK_EQUAL(prop.get(), 40); + BOOST_CHECK_EQUAL(setter._x, 40); + + prop.set(34); + BOOST_CHECK_EQUAL(prop.get(), 32); + BOOST_CHECK_EQUAL(setter._x, 32); +} -- cgit v1.2.3