| // Copyright 2020 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <lib/fdio/fd.h> |
| #include <lib/zx/clock.h> |
| #include <lib/zx/time.h> |
| #include <lib/zx/timer.h> |
| #include <lib/zxio/null.h> |
| #include <lib/zxio/ops.h> |
| #include <sys/timerfd.h> |
| #include <time.h> |
| #include <zircon/assert.h> |
| |
| #include <algorithm> |
| |
| #include <fbl/auto_lock.h> |
| |
| #include "fdio_unistd.h" |
| #include "internal.h" |
| #include "zxio.h" |
| |
| // An implementation of a POSIX timerfd. |
| struct fdio_timer_t { |
| zxio_t io; |
| |
| // The zx::timer object that implements the timerfd. |
| zx::timer handle; |
| |
| mtx_t lock; |
| |
| zx::time current_deadline __TA_GUARDED(lock); |
| zx::duration interval __TA_GUARDED(lock); |
| }; |
| |
| static_assert(sizeof(fdio_timer_t) <= sizeof(zxio_storage_t), |
| "fdio_timer_t must fit inside zxio_storage_t."); |
| |
| static struct timespec duration_to_timespec(zx::duration duration) { |
| struct timespec result = {}; |
| result.tv_sec = duration.to_secs(); |
| result.tv_nsec = duration % zx::sec(1); |
| return result; |
| } |
| |
| static bool timespec_to_duration(const struct timespec* spec, zx::duration* out_duration) { |
| if (!spec || spec->tv_sec < 0 || spec->tv_nsec < 0 || spec->tv_sec > INT64_MAX / ZX_SEC(1)) { |
| return false; |
| } |
| *out_duration = zx::sec(spec->tv_sec) + zx::nsec(spec->tv_nsec); |
| return true; |
| } |
| |
| static zx_status_t fdio_timer_close(zxio_t* io) { |
| auto* timer = reinterpret_cast<fdio_timer_t*>(io); |
| timer->~fdio_timer_t(); |
| return ZX_OK; |
| } |
| |
| static zx_status_t fdio_timer_readv(zxio_t* io, const zx_iovec_t* vector, size_t vector_count, |
| zxio_flags_t flags, size_t* out_actual) { |
| if (fdio_iovec_get_capacity(vector, vector_count) < sizeof(uint64_t)) { |
| return ZX_ERR_BUFFER_TOO_SMALL; |
| } |
| |
| fdio_timer_t* timer = reinterpret_cast<fdio_timer_t*>(io); |
| |
| fbl::AutoLock lock(&timer->lock); |
| if (timer->current_deadline == zx::time()) { |
| // The timer was never set. |
| return ZX_ERR_SHOULD_WAIT; |
| } |
| |
| zx::time now = zx::clock::get_monotonic(); |
| if (timer->current_deadline > now) { |
| return ZX_ERR_SHOULD_WAIT; |
| } |
| |
| uint64_t count = 1; |
| if (timer->interval > zx::duration()) { |
| count = (now - timer->current_deadline) / timer->interval + 1; |
| timer->current_deadline += timer->interval * count; |
| // After reading the current value, the timer will no longer be readable until we reach the next |
| // deadline. Calling zx_timer_set will clear the ZX_TIMER_SIGNALED signal until at least then. |
| zx_status_t status = timer->handle.set(timer->current_deadline, zx::duration()); |
| ZX_ASSERT(status == ZX_OK); |
| } else { |
| timer->current_deadline = zx::time(); |
| // After reading the current value for non-repeating timer, the timer will never produce any |
| // more values, so we use zx_timer_cancel to clear the ZX_TIMER_SIGNALED signal. |
| zx_status_t status = timer->handle.cancel(); |
| ZX_ASSERT(status == ZX_OK); |
| } |
| |
| fdio_iovec_copy_to(reinterpret_cast<const uint8_t*>(&count), sizeof(count), vector, vector_count, |
| out_actual); |
| |
| return ZX_OK; |
| } |
| |
| static void fdio_timer_wait_begin(zxio_t* io, zxio_signals_t zxio_signals, zx_handle_t* out_handle, |
| zx_signals_t* out_zx_signals) { |
| fdio_timer_t* timer = reinterpret_cast<fdio_timer_t*>(io); |
| zx_signals_t zx_signals = ZX_SIGNAL_NONE; |
| if (zxio_signals & ZXIO_SIGNAL_READABLE) { |
| zx_signals |= ZX_TIMER_SIGNALED; |
| } |
| *out_handle = timer->handle.get(); |
| *out_zx_signals = zx_signals; |
| } |
| |
| static void fdio_timer_wait_end(zxio_t* io, zx_signals_t zx_signals, |
| zxio_signals_t* out_zxio_signals) { |
| zxio_signals_t zxio_signals = ZXIO_SIGNAL_NONE; |
| if (zx_signals & ZX_TIMER_SIGNALED) { |
| zxio_signals |= ZXIO_SIGNAL_READABLE; |
| } |
| *out_zxio_signals = zxio_signals; |
| } |
| |
| static constexpr zxio_ops_t fdio_timer_ops = []() { |
| zxio_ops_t ops = zxio_default_ops; |
| ops.close = fdio_timer_close; |
| ops.readv = fdio_timer_readv; |
| ops.wait_begin = fdio_timer_wait_begin; |
| ops.wait_end = fdio_timer_wait_end; |
| return ops; |
| }(); |
| |
| static fdio_timer_t* to_timer(const fdio_ptr& io) { |
| if (!io) { |
| return nullptr; |
| } |
| auto& zxio = io->zxio_storage().io; |
| if (zxio_get_ops(&zxio) != &fdio_timer_ops) { |
| return nullptr; |
| } |
| return reinterpret_cast<fdio_timer_t*>(&zxio); |
| } |
| |
| __EXPORT |
| int timerfd_create(int clockid, int flags) { |
| zx_clock_t zx_clock_id = ZX_CLOCK_MONOTONIC; |
| switch (clockid) { |
| case CLOCK_REALTIME: |
| return ERRNO(ENOSYS); |
| case CLOCK_MONOTONIC: |
| zx_clock_id = ZX_CLOCK_MONOTONIC; |
| break; |
| default: |
| return ERRNO(EINVAL); |
| } |
| |
| if (flags & ~(TFD_CLOEXEC | TFD_NONBLOCK)) { |
| // TODO: Implement TFD_TIMER_ABSTIME. |
| return ERRNO(EINVAL); |
| } |
| |
| zx::timer handle; |
| zx_status_t status = zx::timer::create(ZX_TIMER_SLACK_LATE, zx_clock_id, &handle); |
| if (status != ZX_OK) { |
| return ERROR(status); |
| } |
| |
| zx::status io = fdio_internal::zxio::create(); |
| if (io.is_error()) { |
| return ERROR(status); |
| } |
| auto storage = &io->zxio_storage(); |
| auto timer = new (storage) fdio_timer_t{ |
| .io = storage->io, |
| .handle = std::move(handle), |
| .lock = {}, |
| .current_deadline = {}, |
| .interval = {}, |
| }; |
| zxio_init(&timer->io, &fdio_timer_ops); |
| |
| if (flags & TFD_CLOEXEC) { |
| io->ioflag() |= IOFLAG_CLOEXEC; |
| } |
| |
| if (flags & TFD_NONBLOCK) { |
| io->ioflag() |= IOFLAG_NONBLOCK; |
| } |
| |
| std::optional fd = bind_to_fd(io.value()); |
| if (fd.has_value()) { |
| return fd.value(); |
| } |
| return ERRNO(EMFILE); |
| } |
| |
| static void fdio_timer_get_current_timespec(fdio_timer_t* timer, struct itimerspec* out_timespec) |
| __TA_REQUIRES(timer->lock) { |
| zx::time now = zx::clock::get_monotonic(); |
| if (timer->interval == zx::duration() && timer->current_deadline <= now) { |
| out_timespec->it_value = duration_to_timespec(zx::duration()); |
| } else { |
| // TODO: Is it ok for it_value if the caller is behind in reading a repeating timer? |
| out_timespec->it_value = duration_to_timespec(timer->current_deadline - now); |
| } |
| out_timespec->it_interval = duration_to_timespec(timer->interval); |
| } |
| |
| __EXPORT int timerfd_settime(int fd, int flags, const struct itimerspec* new_value, |
| struct itimerspec* old_value) { |
| fdio_ptr io = fd_to_io(fd); |
| if (io == nullptr) { |
| return ERRNO(EBADF); |
| } |
| |
| fdio_timer_t* timer = to_timer(io); |
| if (!timer) { |
| return ERRNO(EINVAL); |
| } |
| |
| if (flags) { |
| // TODO: Implement TFD_TIMER_ABSTIME. |
| return ERRNO(EINVAL); |
| } |
| |
| zx::duration value; |
| if (!timespec_to_duration(&new_value->it_value, &value)) { |
| return ERRNO(EINVAL); |
| } |
| zx::duration interval; |
| if (!timespec_to_duration(&new_value->it_interval, &interval)) { |
| return ERRNO(EINVAL); |
| } |
| |
| fbl::AutoLock lock(&timer->lock); |
| |
| struct itimerspec old = {}; |
| if (old_value) { |
| fdio_timer_get_current_timespec(timer, &old); |
| } |
| |
| zx::time current_deadline = value.get() == 0 ? zx::time() : zx::deadline_after(value); |
| zx_status_t status = ZX_OK; |
| |
| if (current_deadline > zx::time()) { |
| status = timer->handle.set(current_deadline, zx::duration()); |
| } else { |
| status = timer->handle.cancel(); |
| } |
| if (status != ZX_OK) { |
| return ERROR(status); |
| } |
| |
| timer->current_deadline = current_deadline; |
| timer->interval = interval; |
| |
| if (old_value) { |
| *old_value = old; |
| } |
| return 0; |
| } |
| |
| __EXPORT |
| int timerfd_gettime(int fd, struct itimerspec* curr_value) { |
| fdio_ptr io = fd_to_io(fd); |
| if (io == nullptr) { |
| return ERRNO(EBADF); |
| } |
| |
| fdio_timer_t* timer = to_timer(io); |
| if (!timer) { |
| return ERRNO(EINVAL); |
| } |
| fbl::AutoLock lock(&timer->lock); |
| fdio_timer_get_current_timespec(timer, curr_value); |
| return 0; |
| } |