cregit-Linux how code gets into the kernel

Release 4.11 sound/oss/midibuf.c

Directory: sound/oss
/*
 * sound/oss/midibuf.c
 *
 * Device file manager for /dev/midi#
 */
/*
 * Copyright (C) by Hannu Savolainen 1993-1997
 *
 * OSS/Free for Linux is distributed under the GNU GENERAL PUBLIC LICENSE (GPL)
 * Version 2 (June 1991). See the "COPYING" file distributed with this software
 * for more info.
 */
/*
 * Thomas Sailer   : ioctl code reworked (vmalloc/vfree removed)
 */
#include <linux/stddef.h>
#include <linux/kmod.h>
#include <linux/spinlock.h>
#include <linux/sched/signal.h>


#define MIDIBUF_C

#include "sound_config.h"


/*
 * Don't make MAX_QUEUE_SIZE larger than 4000
 */


#define MAX_QUEUE_SIZE	4000


static wait_queue_head_t midi_sleeper[MAX_MIDI_DEV];

static wait_queue_head_t input_sleeper[MAX_MIDI_DEV];


struct midi_buf
{
	


int len, head, tail;
	
unsigned char queue[MAX_QUEUE_SIZE];
};


struct midi_parms
{
	
long prech_timeout;	/*
                                 * Timeout before the first ch
                                 */
};


static struct midi_buf *midi_out_buf[MAX_MIDI_DEV] = {NULL};

static struct midi_buf *midi_in_buf[MAX_MIDI_DEV] = {NULL};

static struct midi_parms parms[MAX_MIDI_DEV];

static void midi_poll(unsigned long dummy);


static DEFINE_TIMER(poll_timer, midi_poll, 0, 0);


static volatile int open_devs;
static DEFINE_SPINLOCK(lock);


#define DATA_AVAIL(q) (q->len)

#define SPACE_AVAIL(q) (MAX_QUEUE_SIZE - q->len)


#define QUEUE_BYTE(q, data) \
	if (SPACE_AVAIL(q)) \
        { \
          unsigned long flags; \
          spin_lock_irqsave(&lock, flags); \
          q->queue[q->tail] = (data); \
          q->len++; q->tail = (q->tail+1) % MAX_QUEUE_SIZE; \
          spin_unlock_irqrestore(&lock, flags); \
        }


#define REMOVE_BYTE(q, data) \
	if (DATA_AVAIL(q)) \
        { \
          unsigned long flags; \
          spin_lock_irqsave(&lock, flags); \
          data = q->queue[q->head]; \
          q->len--; q->head = (q->head+1) % MAX_QUEUE_SIZE; \
          spin_unlock_irqrestore(&lock, flags); \
        }


static void drain_midi_queue(int dev) { /* * Give the Midi driver time to drain its output queues */ if (midi_devs[dev]->buffer_status != NULL) wait_event_interruptible_timeout(midi_sleeper[dev], !midi_devs[dev]->buffer_status(dev), HZ/10); }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)3784.09%685.71%
Arnd Bergmann715.91%114.29%
Total44100.00%7100.00%


static void midi_input_intr(int dev, unsigned char data) { if (midi_in_buf[dev] == NULL) return; if (data == 0xfe) /* * Active sensing */ return; /* * Ignore */ if (SPACE_AVAIL(midi_in_buf[dev])) { QUEUE_BYTE(midi_in_buf[dev], data); wake_up(&input_sleeper[dev]); } }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)63100.00%3100.00%
Total63100.00%3100.00%


static void midi_output_intr(int dev) { /* * Currently NOP */ }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)10100.00%2100.00%
Total10100.00%2100.00%


static void midi_poll(unsigned long dummy) { unsigned long flags; int dev; spin_lock_irqsave(&lock, flags); if (open_devs) { for (dev = 0; dev < num_midis; dev++) if (midi_devs[dev] != NULL && midi_out_buf[dev] != NULL) { while (DATA_AVAIL(midi_out_buf[dev])) { int ok; int c = midi_out_buf[dev]->queue[midi_out_buf[dev]->head]; spin_unlock_irqrestore(&lock,flags);/* Give some time to others */ ok = midi_devs[dev]->outputc(dev, c); spin_lock_irqsave(&lock, flags); if (!ok) break; midi_out_buf[dev]->head = (midi_out_buf[dev]->head + 1) % MAX_QUEUE_SIZE; midi_out_buf[dev]->len--; } if (DATA_AVAIL(midi_out_buf[dev]) < 100) wake_up(&midi_sleeper[dev]); } poll_timer.expires = (1) + jiffies; add_timer(&poll_timer); /* * Come back later */ } spin_unlock_irqrestore(&lock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)17886.41%571.43%
Peter Wächtler199.22%114.29%
Clemens Ladisch94.37%114.29%
Total206100.00%7100.00%


int MIDIbuf_open(int dev, struct file *file) { int mode, err; dev = dev >> 4; mode = translate_mode(file); if (num_midis > MAX_MIDI_DEV) { printk(KERN_ERR "midi: Too many midi interfaces\n"); num_midis = MAX_MIDI_DEV; } if (dev < 0 || dev >= num_midis || midi_devs[dev] == NULL) return -ENXIO; /* * Interrupts disabled. Be careful */ module_put(midi_devs[dev]->owner); if ((err = midi_devs[dev]->open(dev, mode, midi_input_intr, midi_output_intr)) < 0) return err; parms[dev].prech_timeout = MAX_SCHEDULE_TIMEOUT; midi_in_buf[dev] = vmalloc(sizeof(struct midi_buf)); if (midi_in_buf[dev] == NULL) { printk(KERN_WARNING "midi: Can't allocate buffer\n"); midi_devs[dev]->close(dev); return -EIO; } midi_in_buf[dev]->len = midi_in_buf[dev]->head = midi_in_buf[dev]->tail = 0; midi_out_buf[dev] = vmalloc(sizeof(struct midi_buf)); if (midi_out_buf[dev] == NULL) { printk(KERN_WARNING "midi: Can't allocate buffer\n"); midi_devs[dev]->close(dev); vfree(midi_in_buf[dev]); midi_in_buf[dev] = NULL; return -EIO; } midi_out_buf[dev]->len = midi_out_buf[dev]->head = midi_out_buf[dev]->tail = 0; open_devs++; init_waitqueue_head(&midi_sleeper[dev]); init_waitqueue_head(&input_sleeper[dev]); if (open_devs < 2) /* This was first open */ { poll_timer.expires = 1 + jiffies; add_timer(&poll_timer); /* Start polling */ } return err; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)31599.68%1090.91%
Christoph Hellwig10.32%19.09%
Total316100.00%11100.00%


void MIDIbuf_release(int dev, struct file *file) { int mode; dev = dev >> 4; mode = translate_mode(file); if (dev < 0 || dev >= num_midis || midi_devs[dev] == NULL) return; /* * Wait until the queue is empty */ if (mode != OPEN_READ) { midi_devs[dev]->outputc(dev, 0xfe); /* * Active sensing to shut the * devices */ wait_event_interruptible(midi_sleeper[dev], !DATA_AVAIL(midi_out_buf[dev])); /* * Sync */ drain_midi_queue(dev); /* * Ensure the output queues are empty */ } midi_devs[dev]->close(dev); open_devs--; if (open_devs == 0) del_timer_sync(&poll_timer); vfree(midi_in_buf[dev]); vfree(midi_out_buf[dev]); midi_in_buf[dev] = NULL; midi_out_buf[dev] = NULL; module_put(midi_devs[dev]->owner); }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)13585.44%975.00%
Linus Torvalds159.49%18.33%
Arnd Bergmann74.43%18.33%
Christoph Hellwig10.63%18.33%
Total158100.00%12100.00%


int MIDIbuf_write(int dev, struct file *file, const char __user *buf, int count) { int c, n, i; unsigned char tmp_data; dev = dev >> 4; if (!count) return 0; c = 0; while (c < count) { n = SPACE_AVAIL(midi_out_buf[dev]); if (n == 0) { /* * No space just now. */ if (file->f_flags & O_NONBLOCK) { c = -EAGAIN; goto out; } if (wait_event_interruptible(midi_sleeper[dev], SPACE_AVAIL(midi_out_buf[dev]))) { c = -EINTR; goto out; } n = SPACE_AVAIL(midi_out_buf[dev]); } if (n > (count - c)) n = count - c; for (i = 0; i < n; i++) { /* BROKE BROKE BROKE - CAN'T DO THIS WITH CLI !! */ /* yes, think the same, so I removed the cli() brackets QUEUE_BYTE is protected against interrupts */ if (copy_from_user((char *) &tmp_data, &(buf)[c], 1)) { c = -EFAULT; goto out; } QUEUE_BYTE(midi_out_buf[dev], tmp_data); c++; } } out: return c; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)17882.03%964.29%
Arnaldo Carvalho de Melo2611.98%17.14%
Arnd Bergmann104.61%17.14%
Peter Wächtler10.46%17.14%
Lucas De Marchi10.46%17.14%
Al Viro10.46%17.14%
Total217100.00%14100.00%


int MIDIbuf_read(int dev, struct file *file, char __user *buf, int count) { int n, c = 0; unsigned char tmp_data; dev = dev >> 4; if (!DATA_AVAIL(midi_in_buf[dev])) { /* * No data yet, wait */ if (file->f_flags & O_NONBLOCK) { c = -EAGAIN; goto out; } wait_event_interruptible_timeout(input_sleeper[dev], DATA_AVAIL(midi_in_buf[dev]), parms[dev].prech_timeout); if (signal_pending(current)) c = -EINTR; /* The user is getting restless */ } if (c == 0 && DATA_AVAIL(midi_in_buf[dev])) /* * Got some bytes */ { n = DATA_AVAIL(midi_in_buf[dev]); if (n > count) n = count; c = 0; while (c < n) { char *fixit; REMOVE_BYTE(midi_in_buf[dev], tmp_data); fixit = (char *) &tmp_data; /* BROKE BROKE BROKE */ /* yes removed the cli() brackets again should q->len,tail&head be atomic_t? */ if (copy_to_user(&(buf)[c], fixit, 1)) { c = -EFAULT; goto out; } c++; } } out: return c; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)18485.58%1275.00%
Arnaldo Carvalho de Melo209.30%16.25%
Arnd Bergmann94.19%16.25%
Al Viro10.47%16.25%
Peter Wächtler10.47%16.25%
Total215100.00%16100.00%


int MIDIbuf_ioctl(int dev, struct file *file, unsigned int cmd, void __user *arg) { int val; dev = dev >> 4; if (((cmd >> 8) & 0xff) == 'C') { if (midi_devs[dev]->coproc) /* Coprocessor ioctl */ return midi_devs[dev]->coproc->ioctl(midi_devs[dev]->coproc->devc, cmd, arg, 0); /* printk("/dev/midi%d: No coprocessor for this device\n", dev);*/ return -ENXIO; } else { switch (cmd) { case SNDCTL_MIDI_PRETIME: if (get_user(val, (int __user *)arg)) return -EFAULT; if (val < 0) val = 0; val = (HZ * val) / 10; parms[dev].prech_timeout = val; return put_user(val, (int __user *)arg); default: if (!midi_devs[dev]->ioctl) return -EINVAL; return midi_devs[dev]->ioctl(dev, cmd, arg); } } }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)18597.37%990.00%
Al Viro52.63%110.00%
Total190100.00%10100.00%

/* No kernel lock - fine */
unsigned int MIDIbuf_poll(int dev, struct file *file, poll_table * wait) { unsigned int mask = 0; dev = dev >> 4; /* input */ poll_wait(file, &input_sleeper[dev], wait); if (DATA_AVAIL(midi_in_buf[dev])) mask |= POLLIN | POLLRDNORM; /* output */ poll_wait(file, &midi_sleeper[dev], wait); if (!SPACE_AVAIL(midi_out_buf[dev])) mask |= POLLOUT | POLLWRNORM; return mask; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)94100.00%6100.00%
Total94100.00%6100.00%


int MIDIbuf_avail(int dev) { if (midi_in_buf[dev]) return DATA_AVAIL (midi_in_buf[dev]); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)27100.00%2100.00%
Total27100.00%2100.00%

EXPORT_SYMBOL(MIDIbuf_avail);

Overall Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)155090.43%2765.85%
Arnaldo Carvalho de Melo462.68%12.44%
Arnd Bergmann331.93%12.44%
Peter Wächtler281.63%12.44%
Linus Torvalds150.88%12.44%
Clemens Ladisch90.53%12.44%
Al Viro70.41%12.44%
Ingo Molnar70.41%24.88%
Andrew Morton60.35%12.44%
Adrian Bunk50.29%12.44%
Thomas Gleixner40.23%12.44%
Christoph Hellwig20.12%12.44%
Lucas De Marchi10.06%12.44%
Uwe Zeisberger10.06%12.44%
Total1714100.00%41100.00%
Directory: sound/oss
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.