clockevents: Add direct ktime programming function
authorMartin Schwidefsky <schwidefsky@de.ibm.com>
Tue, 23 Aug 2011 13:29:43 +0000 (15:29 +0200)
committerThomas Gleixner <tglx@linutronix.de>
Thu, 8 Sep 2011 09:10:56 +0000 (11:10 +0200)
There is at least one architecture (s390) with a sane clockevent device
that can be programmed with the equivalent of a ktime. No need to create
a delta against the current time, the ktime can be used directly.

A new clock device function 'set_next_ktime' is introduced that is called
with the unmodified ktime for the timer if the clock event device has the
CLOCK_EVT_FEAT_KTIME bit set.

Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: john stultz <johnstul@us.ibm.com>
Link: http://lkml.kernel.org/r/20110823133142.815350967@de.ibm.com
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
include/linux/clockchips.h
kernel/time/clockevents.c

index 39bb050bdbb211d3f71e8738a317b147b950a3e7..81e803e90aa43ac42a48b678869333fa8c0e4f4e 100644 (file)
@@ -45,20 +45,22 @@ enum clock_event_nofitiers {
  */
 #define CLOCK_EVT_FEAT_PERIODIC                0x000001
 #define CLOCK_EVT_FEAT_ONESHOT         0x000002
+#define CLOCK_EVT_FEAT_KTIME           0x000004
 /*
  * x86(64) specific misfeatures:
  *
  * - Clockevent source stops in C3 State and needs broadcast support.
  * - Local APIC timer is used as a dummy device.
  */
-#define CLOCK_EVT_FEAT_C3STOP          0x000004
-#define CLOCK_EVT_FEAT_DUMMY           0x000008
+#define CLOCK_EVT_FEAT_C3STOP          0x000008
+#define CLOCK_EVT_FEAT_DUMMY           0x000010
 
 /**
  * struct clock_event_device - clock event device descriptor
  * @event_handler:     Assigned by the framework to be called by the low
  *                     level handler of the event source
- * @set_next_event:    set next event function
+ * @set_next_event:    set next event function using a clocksource delta
+ * @set_next_ktime:    set next event function using a direct ktime value
  * @next_event:                local storage for the next event in oneshot mode
  * @max_delta_ns:      maximum delta value in ns
  * @min_delta_ns:      minimum delta value in ns
@@ -81,6 +83,8 @@ struct clock_event_device {
        void                    (*event_handler)(struct clock_event_device *);
        int                     (*set_next_event)(unsigned long evt,
                                                  struct clock_event_device *);
+       int                     (*set_next_ktime)(ktime_t expires,
+                                                 struct clock_event_device *);
        ktime_t                 next_event;
        u64                     max_delta_ns;
        u64                     min_delta_ns;
index 713ef94eceef97349f85366b8c9883e426d1c471..1ecd6ba36d6c6d3d0ce404450878ea5da047e01d 100644 (file)
@@ -216,6 +216,10 @@ int clockevents_program_event(struct clock_event_device *dev, ktime_t expires,
        if (dev->mode == CLOCK_EVT_MODE_SHUTDOWN)
                return 0;
 
+       /* Shortcut for clockevent devices that can deal with ktime. */
+       if (dev->features & CLOCK_EVT_FEAT_KTIME)
+               return dev->set_next_ktime(expires, dev);
+
        delta = ktime_to_ns(ktime_sub(expires, ktime_get()));
        if (delta <= 0)
                return force ? clockevents_program_min_delta(dev) : -ETIME;