]> asedeno.scripts.mit.edu Git - linux.git/blobdiff - sound/core/timer.c
Merge tag 'arc-5.6-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/vgupta/arc
[linux.git] / sound / core / timer.c
index 24fed5c78273934028e392a24d34ef611da43139..d9f85f2d66a3d3abb6e3223c71209d219d9f8355 100644 (file)
@@ -44,6 +44,28 @@ MODULE_PARM_DESC(timer_tstamp_monotonic, "Use posix monotonic clock source for t
 MODULE_ALIAS_CHARDEV(CONFIG_SND_MAJOR, SNDRV_MINOR_TIMER);
 MODULE_ALIAS("devname:snd/timer");
 
+enum timer_tread_format {
+       TREAD_FORMAT_NONE = 0,
+       TREAD_FORMAT_TIME64,
+       TREAD_FORMAT_TIME32,
+};
+
+struct snd_timer_tread32 {
+       int event;
+       s32 tstamp_sec;
+       s32 tstamp_nsec;
+       unsigned int val;
+};
+
+struct snd_timer_tread64 {
+       int event;
+       u8 pad1[4];
+       s64 tstamp_sec;
+       s64 tstamp_nsec;
+       unsigned int val;
+       u8 pad2[4];
+};
+
 struct snd_timer_user {
        struct snd_timer_instance *timeri;
        int tread;              /* enhanced read with timestamps and events */
@@ -55,16 +77,40 @@ struct snd_timer_user {
        int queue_size;
        bool disconnected;
        struct snd_timer_read *queue;
-       struct snd_timer_tread *tqueue;
+       struct snd_timer_tread64 *tqueue;
        spinlock_t qlock;
        unsigned long last_resolution;
        unsigned int filter;
-       struct timespec tstamp;         /* trigger tstamp */
+       struct timespec64 tstamp;               /* trigger tstamp */
        wait_queue_head_t qchange_sleep;
        struct fasync_struct *fasync;
        struct mutex ioctl_lock;
 };
 
+struct snd_timer_status32 {
+       s32 tstamp_sec;                 /* Timestamp - last update */
+       s32 tstamp_nsec;
+       unsigned int resolution;        /* current period resolution in ns */
+       unsigned int lost;              /* counter of master tick lost */
+       unsigned int overrun;           /* count of read queue overruns */
+       unsigned int queue;             /* used queue size */
+       unsigned char reserved[64];     /* reserved */
+};
+
+#define SNDRV_TIMER_IOCTL_STATUS32     _IOR('T', 0x14, struct snd_timer_status32)
+
+struct snd_timer_status64 {
+       s64 tstamp_sec;                 /* Timestamp - last update */
+       s64 tstamp_nsec;
+       unsigned int resolution;        /* current period resolution in ns */
+       unsigned int lost;              /* counter of master tick lost */
+       unsigned int overrun;           /* count of read queue overruns */
+       unsigned int queue;             /* used queue size */
+       unsigned char reserved[64];     /* reserved */
+};
+
+#define SNDRV_TIMER_IOCTL_STATUS64     _IOR('T', 0x14, struct snd_timer_status64)
+
 /* list of timers */
 static LIST_HEAD(snd_timer_list);
 
@@ -453,12 +499,12 @@ static void snd_timer_notify1(struct snd_timer_instance *ti, int event)
        struct snd_timer *timer = ti->timer;
        unsigned long resolution = 0;
        struct snd_timer_instance *ts;
-       struct timespec tstamp;
+       struct timespec64 tstamp;
 
        if (timer_tstamp_monotonic)
-               ktime_get_ts(&tstamp);
+               ktime_get_ts64(&tstamp);
        else
-               getnstimeofday(&tstamp);
+               ktime_get_real_ts64(&tstamp);
        if (snd_BUG_ON(event < SNDRV_TIMER_EVENT_START ||
                       event > SNDRV_TIMER_EVENT_PAUSE))
                return;
@@ -890,7 +936,7 @@ int snd_timer_new(struct snd_card *card, char *id, struct snd_timer_id *tid,
 {
        struct snd_timer *timer;
        int err;
-       static struct snd_device_ops ops = {
+       static const struct snd_device_ops ops = {
                .dev_free = snd_timer_dev_free,
                .dev_register = snd_timer_dev_register,
                .dev_disconnect = snd_timer_dev_disconnect,
@@ -1025,7 +1071,7 @@ static int snd_timer_dev_disconnect(struct snd_device *device)
        return 0;
 }
 
-void snd_timer_notify(struct snd_timer *timer, int event, struct timespec *tstamp)
+void snd_timer_notify(struct snd_timer *timer, int event, struct timespec64 *tstamp)
 {
        unsigned long flags;
        unsigned long resolution = 0;
@@ -1153,7 +1199,7 @@ static int snd_timer_s_close(struct snd_timer *timer)
        return 0;
 }
 
-static struct snd_timer_hardware snd_timer_system =
+static const struct snd_timer_hardware snd_timer_system =
 {
        .flags =        SNDRV_TIMER_HW_FIRST | SNDRV_TIMER_HW_TASKLET,
        .resolution =   1000000000L / HZ,
@@ -1305,7 +1351,7 @@ static void snd_timer_user_interrupt(struct snd_timer_instance *timeri,
 }
 
 static void snd_timer_user_append_to_tqueue(struct snd_timer_user *tu,
-                                           struct snd_timer_tread *tread)
+                                           struct snd_timer_tread64 *tread)
 {
        if (tu->qused >= tu->queue_size) {
                tu->overrun++;
@@ -1318,11 +1364,11 @@ static void snd_timer_user_append_to_tqueue(struct snd_timer_user *tu,
 
 static void snd_timer_user_ccallback(struct snd_timer_instance *timeri,
                                     int event,
-                                    struct timespec *tstamp,
+                                    struct timespec64 *tstamp,
                                     unsigned long resolution)
 {
        struct snd_timer_user *tu = timeri->callback_data;
-       struct snd_timer_tread r1;
+       struct snd_timer_tread64 r1;
        unsigned long flags;
 
        if (event >= SNDRV_TIMER_EVENT_START &&
@@ -1332,7 +1378,8 @@ static void snd_timer_user_ccallback(struct snd_timer_instance *timeri,
                return;
        memset(&r1, 0, sizeof(r1));
        r1.event = event;
-       r1.tstamp = *tstamp;
+       r1.tstamp_sec = tstamp->tv_sec;
+       r1.tstamp_nsec = tstamp->tv_nsec;
        r1.val = resolution;
        spin_lock_irqsave(&tu->qlock, flags);
        snd_timer_user_append_to_tqueue(tu, &r1);
@@ -1354,8 +1401,8 @@ static void snd_timer_user_tinterrupt(struct snd_timer_instance *timeri,
                                      unsigned long ticks)
 {
        struct snd_timer_user *tu = timeri->callback_data;
-       struct snd_timer_tread *r, r1;
-       struct timespec tstamp;
+       struct snd_timer_tread64 *r, r1;
+       struct timespec64 tstamp;
        int prev, append = 0;
 
        memset(&r1, 0, sizeof(r1));
@@ -1368,14 +1415,15 @@ static void snd_timer_user_tinterrupt(struct snd_timer_instance *timeri,
        }
        if (tu->last_resolution != resolution || ticks > 0) {
                if (timer_tstamp_monotonic)
-                       ktime_get_ts(&tstamp);
+                       ktime_get_ts64(&tstamp);
                else
-                       getnstimeofday(&tstamp);
+                       ktime_get_real_ts64(&tstamp);
        }
        if ((tu->filter & (1 << SNDRV_TIMER_EVENT_RESOLUTION)) &&
            tu->last_resolution != resolution) {
                r1.event = SNDRV_TIMER_EVENT_RESOLUTION;
-               r1.tstamp = tstamp;
+               r1.tstamp_sec = tstamp.tv_sec;
+               r1.tstamp_nsec = tstamp.tv_nsec;
                r1.val = resolution;
                snd_timer_user_append_to_tqueue(tu, &r1);
                tu->last_resolution = resolution;
@@ -1389,14 +1437,16 @@ static void snd_timer_user_tinterrupt(struct snd_timer_instance *timeri,
                prev = tu->qtail == 0 ? tu->queue_size - 1 : tu->qtail - 1;
                r = &tu->tqueue[prev];
                if (r->event == SNDRV_TIMER_EVENT_TICK) {
-                       r->tstamp = tstamp;
+                       r->tstamp_sec = tstamp.tv_sec;
+                       r->tstamp_nsec = tstamp.tv_nsec;
                        r->val += ticks;
                        append++;
                        goto __wake;
                }
        }
        r1.event = SNDRV_TIMER_EVENT_TICK;
-       r1.tstamp = tstamp;
+       r1.tstamp_sec = tstamp.tv_sec;
+       r1.tstamp_nsec = tstamp.tv_nsec;
        r1.val = ticks;
        snd_timer_user_append_to_tqueue(tu, &r1);
        append++;
@@ -1411,7 +1461,7 @@ static void snd_timer_user_tinterrupt(struct snd_timer_instance *timeri,
 static int realloc_user_queue(struct snd_timer_user *tu, int size)
 {
        struct snd_timer_read *queue = NULL;
-       struct snd_timer_tread *tqueue = NULL;
+       struct snd_timer_tread64 *tqueue = NULL;
 
        if (tu->tread) {
                tqueue = kcalloc(size, sizeof(*tqueue), GFP_KERNEL);
@@ -1850,11 +1900,11 @@ static int snd_timer_user_params(struct file *file,
        tu->qhead = tu->qtail = tu->qused = 0;
        if (tu->timeri->flags & SNDRV_TIMER_IFLG_EARLY_EVENT) {
                if (tu->tread) {
-                       struct snd_timer_tread tread;
+                       struct snd_timer_tread64 tread;
                        memset(&tread, 0, sizeof(tread));
                        tread.event = SNDRV_TIMER_EVENT_EARLY;
-                       tread.tstamp.tv_sec = 0;
-                       tread.tstamp.tv_nsec = 0;
+                       tread.tstamp_sec = 0;
+                       tread.tstamp_nsec = 0;
                        tread.val = 0;
                        snd_timer_user_append_to_tqueue(tu, &tread);
                } else {
@@ -1875,17 +1925,41 @@ static int snd_timer_user_params(struct file *file,
        return err;
 }
 
-static int snd_timer_user_status(struct file *file,
-                                struct snd_timer_status __user *_status)
+static int snd_timer_user_status32(struct file *file,
+                                  struct snd_timer_status32 __user *_status)
+ {
+       struct snd_timer_user *tu;
+       struct snd_timer_status32 status;
+
+       tu = file->private_data;
+       if (!tu->timeri)
+               return -EBADFD;
+       memset(&status, 0, sizeof(status));
+       status.tstamp_sec = tu->tstamp.tv_sec;
+       status.tstamp_nsec = tu->tstamp.tv_nsec;
+       status.resolution = snd_timer_resolution(tu->timeri);
+       status.lost = tu->timeri->lost;
+       status.overrun = tu->overrun;
+       spin_lock_irq(&tu->qlock);
+       status.queue = tu->qused;
+       spin_unlock_irq(&tu->qlock);
+       if (copy_to_user(_status, &status, sizeof(status)))
+               return -EFAULT;
+       return 0;
+}
+
+static int snd_timer_user_status64(struct file *file,
+                                  struct snd_timer_status64 __user *_status)
 {
        struct snd_timer_user *tu;
-       struct snd_timer_status status;
+       struct snd_timer_status64 status;
 
        tu = file->private_data;
        if (!tu->timeri)
                return -EBADFD;
        memset(&status, 0, sizeof(status));
-       status.tstamp = tu->tstamp;
+       status.tstamp_sec = tu->tstamp.tv_sec;
+       status.tstamp_nsec = tu->tstamp.tv_nsec;
        status.resolution = snd_timer_resolution(tu->timeri);
        status.lost = tu->timeri->lost;
        status.overrun = tu->overrun;
@@ -1960,6 +2034,36 @@ static int snd_timer_user_pause(struct file *file)
        return 0;
 }
 
+static int snd_timer_user_tread(void __user *argp, struct snd_timer_user *tu,
+                               unsigned int cmd, bool compat)
+{
+       int __user *p = argp;
+       int xarg, old_tread;
+
+       if (tu->timeri) /* too late */
+               return -EBUSY;
+       if (get_user(xarg, p))
+               return -EFAULT;
+
+       old_tread = tu->tread;
+
+       if (!xarg)
+               tu->tread = TREAD_FORMAT_NONE;
+       else if (cmd == SNDRV_TIMER_IOCTL_TREAD64 ||
+                (IS_ENABLED(CONFIG_64BIT) && !compat))
+               tu->tread = TREAD_FORMAT_TIME64;
+       else
+               tu->tread = TREAD_FORMAT_TIME32;
+
+       if (tu->tread != old_tread &&
+           realloc_user_queue(tu, tu->queue_size) < 0) {
+               tu->tread = old_tread;
+               return -ENOMEM;
+       }
+
+       return 0;
+}
+
 enum {
        SNDRV_TIMER_IOCTL_START_OLD = _IO('T', 0x20),
        SNDRV_TIMER_IOCTL_STOP_OLD = _IO('T', 0x21),
@@ -1968,7 +2072,7 @@ enum {
 };
 
 static long __snd_timer_user_ioctl(struct file *file, unsigned int cmd,
-                                unsigned long arg)
+                                unsigned long arg, bool compat)
 {
        struct snd_timer_user *tu;
        void __user *argp = (void __user *)arg;
@@ -1980,23 +2084,9 @@ static long __snd_timer_user_ioctl(struct file *file, unsigned int cmd,
                return put_user(SNDRV_TIMER_VERSION, p) ? -EFAULT : 0;
        case SNDRV_TIMER_IOCTL_NEXT_DEVICE:
                return snd_timer_user_next_device(argp);
-       case SNDRV_TIMER_IOCTL_TREAD:
-       {
-               int xarg, old_tread;
-
-               if (tu->timeri) /* too late */
-                       return -EBUSY;
-               if (get_user(xarg, p))
-                       return -EFAULT;
-               old_tread = tu->tread;
-               tu->tread = xarg ? 1 : 0;
-               if (tu->tread != old_tread &&
-                   realloc_user_queue(tu, tu->queue_size) < 0) {
-                       tu->tread = old_tread;
-                       return -ENOMEM;
-               }
-               return 0;
-       }
+       case SNDRV_TIMER_IOCTL_TREAD_OLD:
+       case SNDRV_TIMER_IOCTL_TREAD64:
+               return snd_timer_user_tread(argp, tu, cmd, compat);
        case SNDRV_TIMER_IOCTL_GINFO:
                return snd_timer_user_ginfo(file, argp);
        case SNDRV_TIMER_IOCTL_GPARAMS:
@@ -2009,8 +2099,10 @@ static long __snd_timer_user_ioctl(struct file *file, unsigned int cmd,
                return snd_timer_user_info(file, argp);
        case SNDRV_TIMER_IOCTL_PARAMS:
                return snd_timer_user_params(file, argp);
-       case SNDRV_TIMER_IOCTL_STATUS:
-               return snd_timer_user_status(file, argp);
+       case SNDRV_TIMER_IOCTL_STATUS32:
+               return snd_timer_user_status32(file, argp);
+       case SNDRV_TIMER_IOCTL_STATUS64:
+               return snd_timer_user_status64(file, argp);
        case SNDRV_TIMER_IOCTL_START:
        case SNDRV_TIMER_IOCTL_START_OLD:
                return snd_timer_user_start(file);
@@ -2034,7 +2126,7 @@ static long snd_timer_user_ioctl(struct file *file, unsigned int cmd,
        long ret;
 
        mutex_lock(&tu->ioctl_lock);
-       ret = __snd_timer_user_ioctl(file, cmd, arg);
+       ret = __snd_timer_user_ioctl(file, cmd, arg, false);
        mutex_unlock(&tu->ioctl_lock);
        return ret;
 }
@@ -2050,13 +2142,29 @@ static int snd_timer_user_fasync(int fd, struct file * file, int on)
 static ssize_t snd_timer_user_read(struct file *file, char __user *buffer,
                                   size_t count, loff_t *offset)
 {
+       struct snd_timer_tread64 *tread;
+       struct snd_timer_tread32 tread32;
        struct snd_timer_user *tu;
        long result = 0, unit;
        int qhead;
        int err = 0;
 
        tu = file->private_data;
-       unit = tu->tread ? sizeof(struct snd_timer_tread) : sizeof(struct snd_timer_read);
+       switch (tu->tread) {
+       case TREAD_FORMAT_TIME64:
+               unit = sizeof(struct snd_timer_tread64);
+               break;
+       case TREAD_FORMAT_TIME32:
+               unit = sizeof(struct snd_timer_tread32);
+               break;
+       case TREAD_FORMAT_NONE:
+               unit = sizeof(struct snd_timer_read);
+               break;
+       default:
+               WARN_ONCE(1, "Corrupt snd_timer_user\n");
+               return -ENOTSUPP;
+       }
+
        mutex_lock(&tu->ioctl_lock);
        spin_lock_irq(&tu->qlock);
        while ((long)count - result >= unit) {
@@ -2095,14 +2203,34 @@ static ssize_t snd_timer_user_read(struct file *file, char __user *buffer,
                tu->qused--;
                spin_unlock_irq(&tu->qlock);
 
-               if (tu->tread) {
-                       if (copy_to_user(buffer, &tu->tqueue[qhead],
-                                        sizeof(struct snd_timer_tread)))
+               tread = &tu->tqueue[qhead];
+
+               switch (tu->tread) {
+               case TREAD_FORMAT_TIME64:
+                       if (copy_to_user(buffer, tread,
+                                        sizeof(struct snd_timer_tread64)))
                                err = -EFAULT;
-               } else {
+                       break;
+               case TREAD_FORMAT_TIME32:
+                       memset(&tread32, 0, sizeof(tread32));
+                       tread32 = (struct snd_timer_tread32) {
+                               .event = tread->event,
+                               .tstamp_sec = tread->tstamp_sec,
+                               .tstamp_nsec = tread->tstamp_nsec,
+                               .val = tread->val,
+                       };
+
+                       if (copy_to_user(buffer, &tread32, sizeof(tread32)))
+                               err = -EFAULT;
+                       break;
+               case TREAD_FORMAT_NONE:
                        if (copy_to_user(buffer, &tu->queue[qhead],
                                         sizeof(struct snd_timer_read)))
                                err = -EFAULT;
+                       break;
+               default:
+                       err = -ENOTSUPP;
+                       break;
                }
 
                spin_lock_irq(&tu->qlock);