From 1c18c451f3a48507e876cfdb03c0a6f2187f62cb Mon Sep 17 00:00:00 2001 From: Daniel Scally Date: Wed, 30 Nov 2022 13:31:08 +0000 Subject: lib/timer: Add timer infrastructure Some sources simply fill the buffers passed by the USB subsystem and return them as quickly as possible. Particularly with compressed formats operating at superspeed this rapidly results in unrealistic frame rates. Add infrastructure that allows us to define a specific framerate and introduce blocking calls that constrain those sources to the framerates expected by the host. Reviewed-by: Kieran Bingham Signed-off-by: Daniel Scally --- lib/meson.build | 1 + lib/timer.c | 96 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 97 insertions(+) create mode 100644 lib/timer.c (limited to 'lib') diff --git a/lib/meson.build b/lib/meson.build index f6991d5..a0d67d5 100644 --- a/lib/meson.build +++ b/lib/meson.build @@ -6,6 +6,7 @@ libuvcgadget_sources = files([ 'jpg-source.c', 'stream.c', 'test-source.c', + 'timer.c', 'uvc.c', 'v4l2.c', 'v4l2-source.c', diff --git a/lib/timer.c b/lib/timer.c new file mode 100644 index 0000000..6627656 --- /dev/null +++ b/lib/timer.c @@ -0,0 +1,96 @@ +/* SPDX-License-Identifier: LGPL-2.1-or-later */ +/* + * libuvcgadget timer utils + * + * Copyright (C) 2022 Daniel Scally + * + * Contact: Daniel Scally + */ + +#include +#include +#include +#include +#include +#include + +#include "timer.h" + +struct timer { + int fd; + struct itimerspec settings; +}; + +struct timer *timer_new(void) +{ + struct timer *timer; + + timer = malloc(sizeof(*timer)); + if (!timer) + return NULL; + + memset(timer, 0, sizeof(*timer)); + + timer->fd = timerfd_create(CLOCK_REALTIME, 0); + if (timer->fd < 0) { + fprintf(stderr, "failed to create timer: %s (%d)\n", + strerror(errno), errno); + goto err_free_timer; + } + + return timer; + +err_free_timer: + free(timer); + + return NULL; +} + +void timer_set_fps(struct timer *timer, int fps) +{ + int ns_per_frame = 1000000000 / fps; + + timer->settings.it_value.tv_nsec = ns_per_frame; + timer->settings.it_interval.tv_nsec = ns_per_frame; +} + +int timer_arm(struct timer *timer) +{ + int ret; + + ret = timerfd_settime(timer->fd, 0, &timer->settings, NULL); + if (ret) + fprintf(stderr, "failed to change timer settings: %s (%d)\n", + strerror(errno), errno); + + return ret; +} + +int timer_disarm(struct timer *timer) +{ + static const struct itimerspec disable_settings = { + { 0, 0 }, + { 0, 0 }, + }; + int ret; + + ret = timerfd_settime(timer->fd, 0, &disable_settings, NULL); + if (ret) + fprintf(stderr, "failed to disable timer: %s (%d)\n", + strerror(errno), errno); + + return ret; +} + +void timer_wait(struct timer *timer) +{ + char read_buf[8]; + + read(timer->fd, read_buf, 8); +} + +void timer_destroy(struct timer *timer) +{ + close(timer->fd); + free(timer); +} -- cgit v1.2.3