aboutsummaryrefslogtreecommitdiff
path: root/src/base/platform/unix
diff options
context:
space:
mode:
authorYuqian Yang <crupest@crupest.life>2025-09-04 02:06:20 +0800
committerYuqian Yang <crupest@crupest.life>2025-09-04 02:06:20 +0800
commitac3d88fc043d628979675dc3ef99ba5e1b4d58ca (patch)
tree6e122b7a4e119e3535b57df5cb7d17a8e034f46c /src/base/platform/unix
parent1b197ca9997eb407407e508eb199b1e6fae78926 (diff)
downloadcru-ac3d88fc043d628979675dc3ef99ba5e1b4d58ca.tar.gz
cru-ac3d88fc043d628979675dc3ef99ba5e1b4d58ca.tar.bz2
cru-ac3d88fc043d628979675dc3ef99ba5e1b4d58ca.zip
Add unix event loop.
Diffstat (limited to 'src/base/platform/unix')
-rw-r--r--src/base/platform/unix/EventLoop.cpp92
-rw-r--r--src/base/platform/unix/Timer.cpp5
2 files changed, 92 insertions, 5 deletions
diff --git a/src/base/platform/unix/EventLoop.cpp b/src/base/platform/unix/EventLoop.cpp
new file mode 100644
index 00000000..337d8a44
--- /dev/null
+++ b/src/base/platform/unix/EventLoop.cpp
@@ -0,0 +1,92 @@
+#include "cru/base/platform/unix/EventLoop.h"
+#include "cru/base/Exception.h"
+
+#include <poll.h>
+#include <algorithm>
+#include <chrono>
+#include <thread>
+
+namespace cru::platform::unix {
+int UnixTimerFile::GetReadFd() const { return this->read_fd_; }
+
+UnixEventLoop::UnixEventLoop() : timer_tag_(1) {}
+
+int UnixEventLoop::Run() {
+ running_thread_ = std::this_thread::get_id();
+
+ pollfd poll_fds[1];
+
+ while (!exit_code_) {
+ int poll_timeout = -1;
+
+ auto iter = std::ranges::find_if(timers_, [](const TimerData &timer) {
+ return timer.timeout <= std::chrono::milliseconds::zero();
+ });
+ if (iter != timers_.end()) {
+ auto &timer = *iter;
+ if (timer.repeat) {
+ while (timer.timeout <= std::chrono::milliseconds::zero()) {
+ timer.timeout += timer.original_timeout;
+ timer.action();
+ }
+ } else {
+ auto action = timer.action;
+ timers_.erase(iter);
+ action();
+ }
+ continue;
+ }
+
+ if (!timers_.empty()) {
+ poll_timeout =
+ std::ranges::min_element(timers_, [](const TimerData &left,
+ const TimerData &right) {
+ return left.timeout < right.timeout;
+ })->timeout.count();
+ }
+
+ auto start = std::chrono::steady_clock::now();
+
+ ::poll(poll_fds, sizeof poll_fds / sizeof *poll_fds, poll_timeout);
+
+ // TODO: A Big Implement to handle X events.
+
+ auto end = std::chrono::steady_clock::now();
+ auto time =
+ std::chrono::duration_cast<std::chrono::milliseconds>(end - start);
+
+ for (auto &timer : timers_) {
+ timer.timeout -= time;
+ }
+ }
+
+ return exit_code_.value();
+}
+
+void UnixEventLoop::RequestQuit(int exit_code) {}
+
+int UnixEventLoop::SetTimer(std::function<void()> action,
+ std::chrono::milliseconds timeout, bool repeat) {
+ if (repeat) {
+ if (timeout <= std::chrono::milliseconds::zero()) {
+ throw Exception("Interval must be bigger than 0.");
+ }
+ } else {
+ if (timeout < std::chrono::milliseconds::zero()) {
+ throw Exception("Timeout must be at least 0.");
+ }
+ }
+
+ auto tag = timer_tag_++;
+
+ if (std::this_thread::get_id() == running_thread_) {
+ timers_.push_back(
+ TimerData(tag, std::move(timeout), repeat, std::move(action)));
+ } else {
+ // TODO: Implement
+ }
+
+ return tag;
+}
+
+} // namespace cru::platform::unix
diff --git a/src/base/platform/unix/Timer.cpp b/src/base/platform/unix/Timer.cpp
deleted file mode 100644
index 25264dc8..00000000
--- a/src/base/platform/unix/Timer.cpp
+++ /dev/null
@@ -1,5 +0,0 @@
-#include "cru/base/platform/unix/Timer.h"
-
-namespace cru::platform::unix {
-int UnixTimerFile::GetReadFd() const { return this->read_fd_; }
-} // namespace cru::platform::unix