summaryrefslogtreecommitdiff
path: root/console-client/timer.c
diff options
context:
space:
mode:
authorMarcus Brinkmann <marcus@gnu.org>2002-09-17 12:26:10 +0000
committerMarcus Brinkmann <marcus@gnu.org>2002-09-17 12:26:10 +0000
commit09e69605b16070de8ce317d86ad736d665a58906 (patch)
tree7c0dc29f4fc660a44d2f7d6ec9d099cc4f8301a6 /console-client/timer.c
parentdbe4c7712b652b7ad3126e94c47b00fa2bbb0a05 (diff)
2002-09-17 Marcus Brinkmann <marcus@gnu.org>
* Makefile (prog-subdirs): Add console-client. sutils/ 2002-09-17 Marcus Brinkmann <marcus@gnu.org> * MAKEDEV.sh (mkdev: vcs): New console device. (mkdev: tty[0-9a-f]|tty[0-9][0-9a-f]): Replaced with new rules for tty[1-9][0-9]. utils/ 2002-09-17 Marcus Brinkmann <marcus@gnu.org> * console-ncurses.c: File removed (the ncursesw console client is now a driver in the console-client). * Makefile: Revert 2002-08-22 change: Do not include`../config.make'. (targets) [LIBNCURSES]: Removed. (SRCS) [LIBNCURSES]: Likewise. (HURDLIBS) [LIBNCURSES]: Likewise. (console-ncurses): Target removed. (console-ncurses-CPPFLAGS): Removed. (console-ncurses-LDLIBS): Likewise. console-client/ 2002-09-17 Marcus Brinkmann <marcus@gnu.org> * Makefile, bdf.c, bdf.h, bell.h, console.c, display.h, driver.c, driver.h, generic-speaker.c, input.h, pc-kbd.c, timer.c, timer.h, unicode.h, vga.c, vga-dynacolor.c, vga-dynacolor.h, vga-dynafont.c, vga-dynafont.h, vga-hw.h, vga-support.c, vga-support.h: New file.
Diffstat (limited to 'console-client/timer.c')
-rw-r--r--console-client/timer.c210
1 files changed, 210 insertions, 0 deletions
diff --git a/console-client/timer.c b/console-client/timer.c
new file mode 100644
index 00000000..7bcdeab0
--- /dev/null
+++ b/console-client/timer.c
@@ -0,0 +1,210 @@
+/* timer.c - A timer module for Mach.
+ Copyright (C) 1995,96,2000,02 Free Software Foundation, Inc.
+ Written by Michael I. Bushnell, p/BSG and Marcus Brinkmann.
+
+ This file is part of the GNU Hurd.
+
+ The GNU Hurd is free software; you can redistribute it and/or
+ modify it under the terms of the GNU General Public License as
+ published by the Free Software Foundation; either version 2, or (at
+ your option) any later version.
+
+ The GNU Hurd is distributed in the hope that it will be useful, but
+ WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA. */
+
+#include <errno.h>
+#include <string.h>
+#include <maptime.h>
+
+#include <cthreads.h>
+
+#include "timer.h"
+
+/* The value of fetch_jiffies() at startup. */
+long long timer_root_jiffies;
+
+/* The mapped time. */
+volatile struct mapped_time_value *timer_mapped_time;
+
+
+/* The timer thread. */
+static thread_t timer_thread;
+
+/* The lock protects the timer list TIMERS. */
+static struct mutex timer_lock;
+
+/* A list of all active timers. */
+static struct timer_list *timers;
+
+
+static inline void
+timer_add_internal (struct timer_list *timer)
+{
+ struct timer_list **tp;
+
+ for (tp = &timers; *tp; tp = &(*tp)->next)
+ if ((*tp)->expires > timer->expires)
+ {
+ timer->next = *tp;
+ timer->next->prev = &timer->next;
+ timer->prev = tp;
+ *tp = timer;
+ break;
+ }
+ if (!*tp)
+ {
+ timer->next = 0;
+ timer->prev = tp;
+ *tp = timer;
+ }
+}
+
+
+/* Make the timer thread aware of new timers at the beginning of the
+ list. */
+static inline void
+kick_timer_thread (void)
+{
+ /* XXX This is a whacky notion. */
+ while (!timer_thread)
+ swtch_pri (0);
+
+ if (timer_thread != mach_thread_self ())
+ {
+ thread_suspend (timer_thread);
+ thread_abort (timer_thread);
+ thread_resume (timer_thread);
+ }
+}
+
+/* The timer thread. */
+static int
+timer_function (int this_is_a_pointless_variable_with_a_rather_long_name)
+{
+ mach_port_t recv = mach_reply_port ();
+ int wait = 0;
+
+ timer_thread = mach_thread_self ();
+
+ mutex_lock (&timer_lock);
+ while (1)
+ {
+ int jiff = fetch_jiffies ();
+
+ if (!timers)
+ wait = -1;
+ else if (timers->expires < jiff)
+ wait = 0;
+ else
+ wait = ((timers->expires - jiff) * 1000) / HZ;
+
+ mutex_unlock (&timer_lock);
+ mach_msg (NULL, (MACH_RCV_MSG | MACH_RCV_INTERRUPT
+ | (wait == -1 ? 0 : MACH_RCV_TIMEOUT)),
+ 0, 0, recv, wait, MACH_PORT_NULL);
+ mutex_lock (&timer_lock);
+
+ while (timers && timers->expires < fetch_jiffies ())
+ {
+ struct timer_list *tp;
+
+ tp = timers;
+
+ timers = timers->next;
+ if (timers)
+ timers->prev = &timers;
+
+ tp->next = 0;
+ tp->prev = 0;
+
+ if ((*tp->fnc) (tp->fnc_data))
+ timer_add_internal (tp);
+ }
+ }
+
+ return 0;
+}
+
+
+/* Initialize the timer component. Must be called once at startup. */
+error_t
+timer_init (void)
+{
+ error_t err;
+ struct timeval tp;
+
+ mutex_init (&timer_lock);
+
+ err = maptime_map (0, 0, &timer_mapped_time);
+ if (err)
+ return err;
+
+ maptime_read (timer_mapped_time, &tp);
+
+ timer_root_jiffies = (long long) tp.tv_sec * HZ
+ + ((long long) tp.tv_usec * HZ) / 1000000;
+
+ cthread_detach (cthread_fork ((cthread_fn_t) timer_function, 0));
+ return 0;
+}
+
+
+/* Initialize the timer TIMER. */
+void
+timer_clear (struct timer_list *timer)
+{
+ memset (timer, 0, sizeof (struct timer_list));
+}
+
+/* Add the timer TIMER to the list. */
+void
+timer_add (struct timer_list *timer)
+{
+ mutex_lock (&timer_lock);
+ timer_add_internal (timer);
+
+ if (timers == timer)
+ kick_timer_thread ();
+
+ mutex_unlock (&timer_lock);
+}
+
+/* Remove the timer TIMER from the list. */
+int
+timer_remove (struct timer_list *timer)
+{
+ mutex_lock (&timer_lock);
+ if (timer->prev)
+ {
+ *timer->prev = timer->next;
+ if (timer->next)
+ timer->next->prev = timer->prev;
+
+ timer->next = 0;
+ timer->prev = 0;
+ mutex_unlock (&timer_lock);
+ return 1;
+ }
+ else
+ {
+ mutex_unlock (&timer_lock);
+ return 0;
+ }
+}
+
+/* Change the expiration time of the timer TIMER to EXPIRES. */
+void
+timer_change (struct timer_list *timer, long long expires)
+{
+ /* XXX Should optimize this. */
+ timer_remove (timer);
+ timer->expires = expires;
+ timer_add (timer);
+}
+