OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [drivers/] [sound/] [sound_switch.c] - Rev 1626

Go to most recent revision | Compare with Previous | Blame | View Log

/*
 * sound/sound_switch.c
 *
 * The system call switch handler
 */
/*
 * Copyright (C) by Hannu Savolainen 1993-1996
 *
 * USS/Lite 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.
 */
#include <linux/config.h>
 
 
#include "sound_config.h"
 
struct sbc_device
  {
    int             usecount;
  };
 
static int      in_use = 0;	/* Total # of open devices */
 
/*
 * /dev/sndstatus -device
 */
static char    *status_buf = NULL;
static int      status_len, status_ptr;
static int      status_busy = 0;
 
static int
put_status (char *s)
{
  int             l = strlen (s);
 
  if (status_len + l >= 4000)
    return 0;
 
  memcpy (&status_buf[status_len], s, l);
  status_len += l;
 
  return 1;
}
 
static int
put_status_int (unsigned int val, int radix)
{
  int             l, v;
 
  static char     hx[] = "0123456789abcdef";
  char            buf[11];
 
  if (!val)
    return put_status ("0");
 
  l = 0;
  buf[10] = 0;
 
  while (val)
    {
      v = val % radix;
      val = val / radix;
 
      buf[9 - l] = hx[v];
      l++;
    }
 
  if (status_len + l >= 4000)
    return 0;
 
  memcpy (&status_buf[status_len], &buf[10 - l], l);
  status_len += l;
 
  return 1;
}
 
static void
init_status (void)
{
  /*
   * Write the status information to the status_buf and update status_len.
   * There is a limit of 4000 bytes for the data.
   */
 
  int             i;
 
  status_ptr = 0;
 
#ifdef SOUND_UNAME_A
  put_status ("Sound Driver:" SOUND_VERSION_STRING
	      " (" SOUND_CONFIG_DATE " " SOUND_CONFIG_BY ",\n"
	      SOUND_UNAME_A ")"
	      "\n");
#else
  put_status ("Sound Driver:" SOUND_VERSION_STRING
	      " (" SOUND_CONFIG_DATE " " SOUND_CONFIG_BY "@"
	      SOUND_CONFIG_HOST "." SOUND_CONFIG_DOMAIN ")"
	      "\n");
#endif
 
  put_status ("Kernel: ");
  put_status (system_utsname.sysname);
  put_status (" ");
  put_status (system_utsname.nodename);
  put_status (" ");
  put_status (system_utsname.release);
  put_status (" ");
  put_status (system_utsname.version);
  put_status (" ");
  put_status (system_utsname.machine);
  put_status ("\n");
 
  if (!put_status ("Config options: "))
    return;
  if (!put_status_int (SELECTED_SOUND_OPTIONS, 16))
    return;
 
  if (!put_status ("\n\nInstalled drivers: \n"))
    return;
 
  for (i = 0; i < num_sound_drivers; i++)
    if (sound_drivers[i].card_type != 0)
      {
	if (!put_status ("Type "))
	  return;
	if (!put_status_int (sound_drivers[i].card_type, 10))
	  return;
	if (!put_status (": "))
	  return;
	if (!put_status (sound_drivers[i].name))
	  return;
 
	if (!put_status ("\n"))
	  return;
      }
 
  if (!put_status ("\nCard config: \n"))
    return;
 
  for (i = 0; i < num_sound_cards; i++)
    if (snd_installed_cards[i].card_type != 0)
      {
	int             drv, tmp;
 
	if (!snd_installed_cards[i].enabled)
	  if (!put_status ("("))
	    return;
 
	/*
	 * if (!put_status_int(snd_installed_cards[i].card_type, 10)) return;
	 * if (!put_status (": ")) return;
	 */
 
	if ((drv = snd_find_driver (snd_installed_cards[i].card_type)) != -1)
	  if (!put_status (sound_drivers[drv].name))
	    return;
 
	if (snd_installed_cards[i].config.io_base)
	  {
	    if (!put_status (" at 0x"))
	      return;
	    if (!put_status_int (snd_installed_cards[i].config.io_base, 16))
	      return;
	  }
 
	tmp = snd_installed_cards[i].config.irq;
	if (tmp != 0)
	  {
	    if (!put_status (" irq "))
	      return;
	    if (tmp < 0)
	      tmp = -tmp;
	    if (!put_status_int (tmp, 10))
	      return;
	  }
 
	if (snd_installed_cards[i].config.dma != -1)
	  {
	    if (!put_status (" drq "))
	      return;
	    if (!put_status_int (snd_installed_cards[i].config.dma, 10))
	      return;
	    if (snd_installed_cards[i].config.dma2 != -1)
	      {
		if (!put_status (","))
		  return;
		if (!put_status_int (snd_installed_cards[i].config.dma2, 10))
		  return;
	      }
	  }
 
	if (!snd_installed_cards[i].enabled)
	  if (!put_status (")"))
	    return;
 
	if (!put_status ("\n"))
	  return;
      }
 
  if (!sound_started)
    {
      put_status ("\n\n***** Sound driver not started *****\n\n");
      return;
    }
 
#ifndef CONFIG_AUDIO
  if (!put_status ("\nAudio devices: NOT ENABLED IN CONFIG\n"))
    return;
#else
  if (!put_status ("\nAudio devices:\n"))
    return;
 
  for (i = 0; i < num_audiodevs; i++)
    {
      if (!put_status_int (i, 10))
	return;
      if (!put_status (": "))
	return;
      if (!put_status (audio_devs[i]->name))
	return;
 
      if (audio_devs[i]->flags & DMA_DUPLEX)
	if (!put_status (" (DUPLEX)"))
	  return;
 
      if (!put_status ("\n"))
	return;
    }
#endif
 
#ifndef CONFIG_SEQUENCER
  if (!put_status ("\nSynth devices: NOT ENABLED IN CONFIG\n"))
    return;
#else
  if (!put_status ("\nSynth devices:\n"))
    return;
 
  for (i = 0; i < num_synths; i++)
    {
      if (!put_status_int (i, 10))
	return;
      if (!put_status (": "))
	return;
      if (!put_status (synth_devs[i]->info->name))
	return;
      if (!put_status ("\n"))
	return;
    }
#endif
 
#ifndef CONFIG_MIDI
  if (!put_status ("\nMidi devices: NOT ENABLED IN CONFIG\n"))
    return;
#else
  if (!put_status ("\nMidi devices:\n"))
    return;
 
  for (i = 0; i < num_midis; i++)
    {
      if (!put_status_int (i, 10))
	return;
      if (!put_status (": "))
	return;
      if (!put_status (midi_devs[i]->info.name))
	return;
      if (!put_status ("\n"))
	return;
    }
#endif
 
  if (!put_status ("\nTimers:\n"))
    return;
 
  for (i = 0; i < num_sound_timers; i++)
    {
      if (!put_status_int (i, 10))
	return;
      if (!put_status (": "))
	return;
      if (!put_status (sound_timer_devs[i]->info.name))
	return;
      if (!put_status ("\n"))
	return;
    }
 
  if (!put_status ("\nMixers:\n"))
    return;
 
  for (i = 0; i < num_mixers; i++)
    {
      if (!put_status_int (i, 10))
	return;
      if (!put_status (": "))
	return;
      if (!put_status (mixer_devs[i]->name))
	return;
      if (!put_status ("\n"))
	return;
    }
}
 
static int
read_status (char *buf, int count)
{
  /*
   * Return at most 'count' bytes from the status_buf.
   */
  int             l, c;
 
  l = count;
  c = status_len - status_ptr;
 
  if (l > c)
    l = c;
  if (l <= 0)
    return 0;
 
  memcpy_tofs (&(buf)[0], &status_buf[status_ptr], l);
  status_ptr += l;
 
  return l;
}
 
int
sound_read_sw (int dev, struct fileinfo *file, char *buf, int count)
{
  DEB (printk ("sound_read_sw(dev=%d, count=%d)\n", dev, count));
 
  switch (dev & 0x0f)
    {
    case SND_DEV_STATUS:
      return read_status (buf, count);
      break;
 
#ifdef CONFIG_AUDIO
    case SND_DEV_DSP:
    case SND_DEV_DSP16:
    case SND_DEV_AUDIO:
      return audio_read (dev, file, buf, count);
      break;
#endif
 
#ifdef CONFIG_SEQUENCER
    case SND_DEV_SEQ:
    case SND_DEV_SEQ2:
      return sequencer_read (dev, file, buf, count);
      break;
#endif
 
#ifdef CONFIG_MIDI
    case SND_DEV_MIDIN:
      return MIDIbuf_read (dev, file, buf, count);
#endif
 
    default:
      printk ("Sound: Undefined minor device %d\n", dev);
    }
 
  return -(EPERM);
}
 
int
sound_write_sw (int dev, struct fileinfo *file, const char *buf, int count)
{
 
  DEB (printk ("sound_write_sw(dev=%d, count=%d)\n", dev, count));
 
  switch (dev & 0x0f)
    {
 
#ifdef CONFIG_SEQUENCER
    case SND_DEV_SEQ:
    case SND_DEV_SEQ2:
      return sequencer_write (dev, file, buf, count);
      break;
#endif
 
#ifdef CONFIG_AUDIO
    case SND_DEV_DSP:
    case SND_DEV_DSP16:
    case SND_DEV_AUDIO:
      return audio_write (dev, file, buf, count);
      break;
#endif
 
#ifdef CONFIG_MIDI
    case SND_DEV_MIDIN:
      return MIDIbuf_write (dev, file, buf, count);
#endif
 
    }
 
  return -(EPERM);
}
 
int
sound_open_sw (int dev, struct fileinfo *file)
{
  int             retval;
 
  DEB (printk ("sound_open_sw(dev=%d)\n", dev));
 
  if ((dev >= SND_NDEVS) || (dev < 0))
    {
      printk ("Invalid minor device %d\n", dev);
      return -(ENXIO);
    }
 
  switch (dev & 0x0f)
    {
    case SND_DEV_STATUS:
      if (status_busy)
	return -(EBUSY);
      status_busy = 1;
      if ((status_buf = (char *) vmalloc (4000)) == NULL)
	return -(EIO);
      status_len = status_ptr = 0;
      init_status ();
      break;
 
    case SND_DEV_CTL:
      if ((dev & 0xf0) && ((dev & 0xf0) >> 4) >= num_mixers)
	return -(ENXIO);
      return 0;
      break;
 
#ifdef CONFIG_SEQUENCER
    case SND_DEV_SEQ:
    case SND_DEV_SEQ2:
      if ((retval = sequencer_open (dev, file)) < 0)
	return retval;
      break;
#endif
 
#ifdef CONFIG_MIDI
    case SND_DEV_MIDIN:
      if ((retval = MIDIbuf_open (dev, file)) < 0)
	return retval;
      break;
#endif
 
#ifdef CONFIG_AUDIO
    case SND_DEV_DSP:
    case SND_DEV_DSP16:
    case SND_DEV_AUDIO:
      if ((retval = audio_open (dev, file)) < 0)
	return retval;
      break;
#endif
 
    default:
      printk ("Invalid minor device %d\n", dev);
      return -(ENXIO);
    }
 
  in_use++;
 
  return 0;
}
 
void
sound_release_sw (int dev, struct fileinfo *file)
{
 
  DEB (printk ("sound_release_sw(dev=%d)\n", dev));
 
  switch (dev & 0x0f)
    {
    case SND_DEV_STATUS:
      if (status_buf)
	vfree (status_buf);
      status_buf = NULL;
      status_busy = 0;
      break;
 
    case SND_DEV_CTL:
      break;
 
#ifdef CONFIG_SEQUENCER
    case SND_DEV_SEQ:
    case SND_DEV_SEQ2:
      sequencer_release (dev, file);
      break;
#endif
 
#ifdef CONFIG_MIDI
    case SND_DEV_MIDIN:
      MIDIbuf_release (dev, file);
      break;
#endif
 
#ifdef CONFIG_AUDIO
    case SND_DEV_DSP:
    case SND_DEV_DSP16:
    case SND_DEV_AUDIO:
      audio_release (dev, file);
      break;
#endif
 
    default:
      printk ("Sound error: Releasing unknown device 0x%02x\n", dev);
    }
  in_use--;
}
 
static int
get_mixer_info (int dev, caddr_t arg)
{
  mixer_info      info;
 
  if (dev < 0 || dev >= num_mixers)
    return -(ENXIO);
 
  strcpy (info.id, mixer_devs[dev]->id);
  strcpy (info.name, mixer_devs[dev]->name);
 
  memcpy_tofs (&((char *) arg)[0], (char *) &info, sizeof (info));
  return 0;
}
 
int
sound_ioctl_sw (int dev, struct fileinfo *file,
		unsigned int cmd, caddr_t arg)
{
  DEB (printk ("sound_ioctl_sw(dev=%d, cmd=0x%x, arg=0x%x)\n", dev, cmd, arg));
 
  if (((cmd >> 8) & 0xff) == 'M' && num_mixers > 0)	/* Mixer ioctl */
    if ((dev & 0x0f) != SND_DEV_CTL)
      {
	int             dtype = dev & 0x0f;
	int             mixdev;
 
	switch (dtype)
	  {
#ifdef CONFIG_AUDIO
	  case SND_DEV_DSP:
	  case SND_DEV_DSP16:
	  case SND_DEV_AUDIO:
	    mixdev = audio_devs[dev >> 4]->mixer_dev;
	    if (mixdev < 0 || mixdev >= num_mixers)
	      return -(ENXIO);
	    if (cmd == SOUND_MIXER_INFO)
	      return get_mixer_info (mixdev, arg);
	    return mixer_devs[mixdev]->ioctl (mixdev, cmd, arg);
	    break;
#endif
 
	  default:
	    if (cmd == SOUND_MIXER_INFO)
	      return get_mixer_info (0, arg);
	    return mixer_devs[0]->ioctl (0, cmd, arg);
	  }
      }
 
  switch (dev & 0x0f)
    {
 
    case SND_DEV_CTL:
 
      if (!num_mixers)
	return -(ENXIO);
 
      dev = dev >> 4;
 
      if (dev >= num_mixers)
	return -(ENXIO);
 
      if (cmd == SOUND_MIXER_INFO)
	return get_mixer_info (dev, arg);
      return mixer_devs[dev]->ioctl (dev, cmd, arg);
      break;
 
#ifdef CONFIG_SEQUENCER
    case SND_DEV_SEQ:
    case SND_DEV_SEQ2:
      return sequencer_ioctl (dev, file, cmd, arg);
      break;
#endif
 
#ifdef CONFIG_AUDIO
    case SND_DEV_DSP:
    case SND_DEV_DSP16:
    case SND_DEV_AUDIO:
      return audio_ioctl (dev, file, cmd, arg);
      break;
#endif
 
#ifdef CONFIG_MIDI
    case SND_DEV_MIDIN:
      return MIDIbuf_ioctl (dev, file, cmd, arg);
      break;
#endif
 
    }
 
  return -(EPERM);
}
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.