summaryrefslogtreecommitdiff
path: root/asl/base
diff options
context:
space:
mode:
Diffstat (limited to 'asl/base')
-rw-r--r--asl/base/BUILD.bazel2
-rw-r--r--asl/base/defer.hpp50
-rw-r--r--asl/base/defer_tests.cpp32
-rw-r--r--asl/base/utility.hpp3
4 files changed, 87 insertions, 0 deletions
diff --git a/asl/base/BUILD.bazel b/asl/base/BUILD.bazel
index 317c20b..3dc715b 100644
--- a/asl/base/BUILD.bazel
+++ b/asl/base/BUILD.bazel
@@ -4,6 +4,7 @@ cc_library(
"annotations.hpp",
"assert.hpp",
"config.hpp",
+ "defer.hpp",
"float.hpp",
"functional.hpp",
"integers.hpp",
@@ -28,6 +29,7 @@ cc_library(
"//asl/types:box",
],
) for name in [
+ "defer",
"float",
"functional",
"integers",
diff --git a/asl/base/defer.hpp b/asl/base/defer.hpp
new file mode 100644
index 0000000..b6d52af
--- /dev/null
+++ b/asl/base/defer.hpp
@@ -0,0 +1,50 @@
+#pragma once
+
+#include "asl/base/utility.hpp"
+#include "asl/base/functional.hpp"
+
+namespace asl
+{
+
+// @Todo Add invokable check
+template<typename Callback>
+class DeferCallback
+{
+ Callback m_callback;
+ bool m_moved = false;
+
+public:
+ template<typename T>
+ explicit DeferCallback(T&& callback) : m_callback(ASL_FWD(callback))
+ {
+ }
+
+ ASL_DELETE_COPY(DeferCallback);
+
+ DeferCallback(DeferCallback&& other) :
+ m_callback(ASL_MOVE(other.m_callback)), m_moved(exchange(other.m_moved, true))
+ {
+ }
+
+ DeferCallback& operator=(DeferCallback&&) = delete;
+
+ ~DeferCallback()
+ {
+ if (!m_moved) { invoke(m_callback); }
+ }
+};
+
+struct DeferFactory
+{
+ // @Todo Add invokable check
+ template<typename Callback>
+ DeferCallback<Callback> operator<<(Callback&& callback) const
+ {
+ return DeferCallback<Callback>(ASL_FWD(callback));
+ }
+};
+
+} // namespace asl
+
+#define ASL_DEFER auto ASL_CONCAT(_defer_, __COUNTER__) = ::asl::DeferFactory{} <<
+
diff --git a/asl/base/defer_tests.cpp b/asl/base/defer_tests.cpp
new file mode 100644
index 0000000..b5139d5
--- /dev/null
+++ b/asl/base/defer_tests.cpp
@@ -0,0 +1,32 @@
+#include "asl/base/defer.hpp"
+#include "asl/testing/testing.hpp"
+
+ASL_TEST(defer)
+{
+ uint32_t a = 0;
+
+ {
+ ASL_DEFER [&a]() { a |= 1; };
+ ASL_TEST_EXPECT(a == 0);
+
+ {
+ ASL_DEFER [&a]() { a |= 2; };
+ ASL_DEFER [&a]() { a |= 4; };
+ ASL_TEST_EXPECT(a == 0);
+ }
+
+ ASL_TEST_EXPECT(a == 6);
+
+ {
+ ASL_DEFER [&a]() { a |= 8; };
+ ASL_TEST_EXPECT(a == 6);
+ }
+
+ ASL_TEST_EXPECT(a == 14);
+
+ ASL_DEFER [&a]() { a |= 16; };
+ ASL_TEST_EXPECT(a == 14);
+ }
+
+ ASL_TEST_EXPECT(a == 31);
+}
diff --git a/asl/base/utility.hpp b/asl/base/utility.hpp
index c03554f..b3fb36b 100644
--- a/asl/base/utility.hpp
+++ b/asl/base/utility.hpp
@@ -92,4 +92,7 @@ constexpr bool is_pow2(isize_t v)
ASL_DEFAULT_COPY(T) \
ASL_DEFAULT_MOVE(T)
+#define ASL_CONCAT2(A, B) A##B
+#define ASL_CONCAT(A, B) ASL_CONCAT2(A, B)
+
} // namespace asl