4  * The default timer for the Level 2 sequencer interface
 
   5  * Uses the (1/HZ sec) timer of kernel.
 
   8  * Copyright (C) by Hannu Savolainen 1993-1997
 
  10  * OSS/Free for Linux is distributed under the GNU GENERAL PUBLIC LICENSE (GPL)
 
  11  * Version 2 (June 1991). See the "COPYING" file distributed with this software
 
  15  * Thomas Sailer   : ioctl code reworked (vmalloc/vfree removed)
 
  16  * Andrew Veliath  : adapted tmr2ticks from level 1 sequencer (avoid overflow)
 
  18 #include <linux/spinlock.h>
 
  19 #include "sound_config.h"
 
  21 static volatile int opened, tmr_running;
 
  22 static volatile time_t tmr_offs, tmr_ctr;
 
  23 static volatile unsigned long ticks_offs;
 
  24 static volatile int curr_tempo, curr_timebase;
 
  25 static volatile unsigned long curr_ticks;
 
  26 static volatile unsigned long next_event_time;
 
  27 static unsigned long prev_event_time;
 
  29 static void     poll_def_tmr(unsigned long dummy);
 
  30 static DEFINE_SPINLOCK(lock);
 
  31 static DEFINE_TIMER(def_tmr, poll_def_tmr, 0, 0);
 
  34 tmr2ticks(int tmr_value)
 
  37          *    Convert timer ticks to MIDI ticks
 
  43         /* tmr_value (ticks per sec) *
 
  44            1000000 (usecs per sec) / HZ (ticks per sec) -=> usecs */
 
  45         tmp = tmr_value * (1000000 / HZ);
 
  46         scale = (60 * 1000000) / (curr_tempo * curr_timebase);  /* usecs per MIDI tick */
 
  47         return (tmp + scale / 2) / scale;
 
  51 poll_def_tmr(unsigned long dummy)
 
  58                           def_tmr.expires = (1) + jiffies;
 
  66                             curr_ticks = ticks_offs + tmr2ticks(tmr_ctr);
 
  68                             if (curr_ticks >= next_event_time)
 
  70                                       next_event_time = (unsigned long) -1;
 
  83         spin_lock_irqsave(&lock,flags);
 
  87         next_event_time = (unsigned long) -1;
 
  90         spin_unlock_irqrestore(&lock,flags);
 
  94 def_tmr_open(int dev, int mode)
 
 107                 def_tmr.expires = (1) + jiffies;
 
 115 def_tmr_close(int dev)
 
 117         opened = tmr_running = 0;
 
 122 def_tmr_event(int dev, unsigned char *event)
 
 124         unsigned char   cmd = event[1];
 
 125         unsigned long   parm = *(int *) &event[4];
 
 130                   parm += prev_event_time;
 
 136                             if (parm <= curr_ticks)     /* It's the time */
 
 137                                     return TIMER_NOT_ARMED;
 
 140                             next_event_time = prev_event_time = time;
 
 167                             ticks_offs += tmr2ticks(tmr_ctr);
 
 174                   seq_copy_to_input(event, 8);
 
 180         return TIMER_NOT_ARMED;
 
 184 def_tmr_get_time(int dev)
 
 192 /* same as sound_timer.c:timer_ioctl!? */
 
 193 static int def_tmr_ioctl(int dev, unsigned int cmd, void __user *arg)
 
 199         case SNDCTL_TMR_SOURCE:
 
 200                 return __put_user(TMR_INTERNAL, p);
 
 202         case SNDCTL_TMR_START:
 
 207         case SNDCTL_TMR_STOP:
 
 211         case SNDCTL_TMR_CONTINUE:
 
 215         case SNDCTL_TMR_TIMEBASE:
 
 216                 if (__get_user(val, p))
 
 225                 return __put_user(curr_timebase, p);
 
 227         case SNDCTL_TMR_TEMPO:
 
 228                 if (__get_user(val, p))
 
 236                         ticks_offs += tmr2ticks(tmr_ctr);
 
 241                 return __put_user(curr_tempo, p);
 
 243         case SNDCTL_SEQ_CTRLRATE:
 
 244                 if (__get_user(val, p))
 
 246                 if (val != 0)   /* Can't change */
 
 248                 val = ((curr_tempo * curr_timebase) + 30) / 60;
 
 249                 return __put_user(val, p);
 
 251         case SNDCTL_SEQ_GETTIME:
 
 252                 return __put_user(curr_ticks, p);
 
 254         case SNDCTL_TMR_METRONOME:
 
 264 def_tmr_arm(int dev, long time)
 
 267                 time = curr_ticks + 1;
 
 268         else if (time <= curr_ticks)    /* It's the time */
 
 271         next_event_time = prev_event_time = time;
 
 276 struct sound_timer_operations default_sound_timer =
 
 278         .owner          = THIS_MODULE,
 
 279         .info           = {"System clock", 0},
 
 280         .priority       = 0,    /* Priority */
 
 281         .devlink        = 0,    /* Local device link */
 
 282         .open           = def_tmr_open,
 
 283         .close          = def_tmr_close,
 
 284         .event          = def_tmr_event,
 
 285         .get_time       = def_tmr_get_time,
 
 286         .ioctl          = def_tmr_ioctl,
 
 287         .arm_timer      = def_tmr_arm