summaryrefslogtreecommitdiff
path: root/include/scope.h
diff options
context:
space:
mode:
authorEugene Kosov <eugene.kosov@mariadb.com>2020-06-18 17:26:01 +0300
committerEugene Kosov <claprix@yandex.ru>2021-12-03 16:12:27 +0600
commit0dfe5a31573cca56b9c642762a3231d1c3b5c08e (patch)
tree1cab5b9b2f1464ead36a88cbb0e7b31fba273c26 /include/scope.h
parent1e54a9716d2d41a059cd6f91a2f7bf6d46e48090 (diff)
downloadmariadb-git-bb-10.7-MDEV-22441-scoped-variable.tar.gz
MDEV-22441 implement a generic way to change a value of a variable in a scopebb-10.7-MDEV-22441-scoped-variable
Example: { auto _= make_scope_value(var, tmp_value); } make_scope_value(): a function which returns RAII object which temporary changes a value of a variable detail::Scope_value: actual implementation of such RAII class. It shouldn't be used directly! That's why it's inside a namespace detail.
Diffstat (limited to 'include/scope.h')
-rw-r--r--include/scope.h45
1 files changed, 45 insertions, 0 deletions
diff --git a/include/scope.h b/include/scope.h
index e0e9fc62969..836c4f77b35 100644
--- a/include/scope.h
+++ b/include/scope.h
@@ -72,3 +72,48 @@ make_scope_exit(Callable &&f)
#define ANONYMOUS_VARIABLE CONCAT(_anonymous_variable, __LINE__)
#define SCOPE_EXIT auto ANONYMOUS_VARIABLE= make_scope_exit
+
+namespace detail
+{
+
+template <typename T> class Scope_value
+{
+public:
+ Scope_value(T &variable, const T &scope_value)
+ : variable_(variable), saved_value_(variable)
+ {
+ variable= scope_value;
+ }
+
+ Scope_value(Scope_value &&rhs)
+ : variable_(rhs.variable_), saved_value_(rhs.saved_value_),
+ engaged_(rhs.engaged_)
+ {
+ rhs.engaged_= false;
+ }
+
+ Scope_value(const Scope_value &)= delete;
+ Scope_value &operator=(const Scope_value &)= delete;
+ Scope_value &operator=(Scope_value &&)= delete;
+
+ ~Scope_value()
+ {
+ if (engaged_)
+ variable_= saved_value_;
+ }
+
+private:
+ T &variable_;
+ T saved_value_;
+ bool engaged_= true;
+};
+
+} // namespace detail
+
+// Use like this:
+// auto _= make_scope_value(var, tmp_value);
+template <typename T>
+detail::Scope_value<T> make_scope_value(T &variable, const T &scope_value)
+{
+ return detail::Scope_value<T>(variable, scope_value);
+}