aboutsummaryrefslogtreecommitdiffstats
path: root/libdevcore
diff options
context:
space:
mode:
authorchriseth <chris@ethereum.org>2018-10-10 22:01:20 +0800
committerGitHub <noreply@github.com>2018-10-10 22:01:20 +0800
commitb5399a7aa03c2575d5aeea3ce55a34486472d639 (patch)
tree36eba9ab4bc25f169e7449a3584904103389a452 /libdevcore
parent5b5a4d0172bb0d3cd9fb49aaf23fa8cbb63548c0 (diff)
parent3f216bac5e4ecd5723df815d13cae7ede4ad21dc (diff)
downloaddexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.tar
dexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.tar.gz
dexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.tar.bz2
dexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.tar.lz
dexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.tar.xz
dexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.tar.zst
dexon-solidity-b5399a7aa03c2575d5aeea3ce55a34486472d639.zip
Merge pull request #5132 from ethereum/genericVisitor
Generic visitor.
Diffstat (limited to 'libdevcore')
-rw-r--r--libdevcore/Visitor.h128
1 files changed, 128 insertions, 0 deletions
diff --git a/libdevcore/Visitor.h b/libdevcore/Visitor.h
new file mode 100644
index 00000000..4030c928
--- /dev/null
+++ b/libdevcore/Visitor.h
@@ -0,0 +1,128 @@
+/*
+ This file is part of solidity.
+
+ solidity 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.
+
+ solidity 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 solidity. If not, see <http://www.gnu.org/licenses/>.
+*/
+/**
+ * Visitor templates.
+ */
+
+#pragma once
+
+#include <functional>
+#include <boost/variant/static_visitor.hpp>
+
+namespace dev
+{
+
+/// Generic visitor used as follows:
+/// boost::apply_visitor(GenericVisitor<Class1, Class2>(
+/// [](Class1& _c) { _c.f(); },
+/// [](Class2& _c) { _c.g(); }
+/// ), variant);
+/// This one does not have a fallback and will fail at
+/// compile-time if you do not specify all variants.
+
+template <class...>
+struct GenericVisitor{};
+
+template <class Visitable, class... Others>
+struct GenericVisitor<Visitable, Others...>: public GenericVisitor<Others...>
+{
+ using GenericVisitor<Others...>::operator ();
+ explicit GenericVisitor(
+ std::function<void(Visitable&)> _visitor,
+ std::function<void(Others&)>... _otherVisitors
+ ):
+ GenericVisitor<Others...>(std::move(_otherVisitors)...),
+ m_visitor(std::move(_visitor))
+ {}
+
+ void operator()(Visitable& _v) const { m_visitor(_v); }
+
+ std::function<void(Visitable&)> m_visitor;
+};
+template <>
+struct GenericVisitor<>: public boost::static_visitor<> {
+ void operator()() const {}
+};
+
+/// Generic visitor with fallback:
+/// boost::apply_visitor(GenericFallbackVisitor<Class1, Class2>(
+/// [](Class1& _c) { _c.f(); },
+/// [](Class2& _c) { _c.g(); }
+/// ), variant);
+/// This one DOES have a fallback and will NOT fail at
+/// compile-time if you do not specify all variants.
+
+template <class...>
+struct GenericFallbackVisitor{};
+
+template <class Visitable, class... Others>
+struct GenericFallbackVisitor<Visitable, Others...>: public GenericFallbackVisitor<Others...>
+{
+ explicit GenericFallbackVisitor(
+ std::function<void(Visitable&)> _visitor,
+ std::function<void(Others&)>... _otherVisitors
+ ):
+ GenericFallbackVisitor<Others...>(std::move(_otherVisitors)...),
+ m_visitor(std::move(_visitor))
+ {}
+
+ using GenericFallbackVisitor<Others...>::operator ();
+ void operator()(Visitable& _v) const { m_visitor(_v); }
+
+ std::function<void(Visitable&)> m_visitor;
+};
+template <>
+struct GenericFallbackVisitor<>: public boost::static_visitor<> {
+ template <class T>
+ void operator()(T&) const { }
+};
+
+/// Generic visitor with fallback that can return a value:
+/// boost::apply_visitor(GenericFallbackReturnsVisitor<ReturnType, Class1, Class2>(
+/// [](Class1& _c) { return _c.f(); },
+/// [](Class2& _c) { return _c.g(); }
+/// ), variant);
+/// This one DOES have a fallback and will NOT fail at
+/// compile-time if you do not specify all variants.
+/// The fallback {}-constructs the return value.
+
+template <class R, class...>
+struct GenericFallbackReturnsVisitor{};
+
+template <class R, class Visitable, class... Others>
+struct GenericFallbackReturnsVisitor<R, Visitable, Others...>: public GenericFallbackReturnsVisitor<R, Others...>
+{
+ explicit GenericFallbackReturnsVisitor(
+ std::function<R(Visitable&)> _visitor,
+ std::function<R(Others&)>... _otherVisitors
+ ):
+ GenericFallbackReturnsVisitor<R, Others...>(std::move(_otherVisitors)...),
+ m_visitor(std::move(_visitor))
+ {}
+
+ using GenericFallbackReturnsVisitor<R, Others...>::operator ();
+ R operator()(Visitable& _v) const { return m_visitor(_v); }
+
+ std::function<R(Visitable&)> m_visitor;
+};
+template <class R>
+struct GenericFallbackReturnsVisitor<R>: public boost::static_visitor<R> {
+ template <class T>
+ R operator()(T&) const { return {}; }
+};
+
+}