pas2_mixer.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:8k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.  * sound/pas2_mixer.c
  3.  *
  4.  * Mixer routines for the Pro Audio Spectrum cards.
  5.  */
  6. /*
  7.  * Copyright (C) by Hannu Savolainen 1993-1997
  8.  *
  9.  * OSS/Free for Linux is distributed under the GNU GENERAL PUBLIC LICENSE (GPL)
  10.  * Version 2 (June 1991). See the "COPYING" file distributed with this software
  11.  * for more info.
  12.  */
  13. /*
  14.  * Thomas Sailer   : ioctl code reworked (vmalloc/vfree removed)
  15.  * Bartlomiej Zolnierkiewicz : added __init to pas_init_mixer()
  16.  */
  17. #include <linux/init.h>
  18. #include "sound_config.h"
  19. #include "pas2.h"
  20. #ifndef DEB
  21. #define DEB(what) /* (what) */
  22. #endif
  23. extern int      translate_code;
  24. extern char     pas_model;
  25. extern int     *pas_osp;
  26. extern int      pas_audiodev;
  27. static int      rec_devices = (SOUND_MASK_MIC); /* Default recording source */
  28. static int      mode_control = 0;
  29. #define POSSIBLE_RECORDING_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_SPEAKER | SOUND_MASK_LINE | SOUND_MASK_MIC | 
  30.  SOUND_MASK_CD | SOUND_MASK_ALTPCM)
  31. #define SUPPORTED_MIXER_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_PCM | SOUND_MASK_SPEAKER | SOUND_MASK_LINE | SOUND_MASK_MIC | 
  32.  SOUND_MASK_CD | SOUND_MASK_ALTPCM | SOUND_MASK_IMIX | 
  33.  SOUND_MASK_VOLUME | SOUND_MASK_BASS | SOUND_MASK_TREBLE | SOUND_MASK_RECLEV)
  34. static int     *levels;
  35. static int      default_levels[32] =
  36. {
  37. 0x3232, /* Master Volume */
  38. 0x3232, /* Bass */
  39. 0x3232, /* Treble */
  40. 0x5050, /* FM */
  41. 0x4b4b, /* PCM */
  42. 0x3232, /* PC Speaker */
  43. 0x4b4b, /* Ext Line */
  44. 0x4b4b, /* Mic */
  45. 0x4b4b, /* CD */
  46. 0x6464, /* Recording monitor */
  47. 0x4b4b, /* SB PCM */
  48. 0x6464 /* Recording level */
  49. };
  50. void
  51. mix_write(unsigned char data, int ioaddr)
  52. {
  53. /*
  54.  * The Revision D cards have a problem with their MVA508 interface. The
  55.  * kludge-o-rama fix is to make a 16-bit quantity with identical LSB and
  56.  * MSBs out of the output byte and to do a 16-bit out to the mixer port -
  57.  * 1. We need to do this because it isn't timing problem but chip access
  58.  * sequence problem.
  59.  */
  60. if (pas_model == 4)
  61.   {
  62.   outw(data | (data << 8), (ioaddr + translate_code) - 1);
  63.   outb((0x80), 0);
  64. } else
  65. pas_write(data, ioaddr);
  66. }
  67. static int
  68. mixer_output(int right_vol, int left_vol, int div, int bits,
  69.      int mixer) /* Input or output mixer */
  70. {
  71. int             left = left_vol * div / 100;
  72. int             right = right_vol * div / 100;
  73. if (bits & 0x10)
  74.   {
  75.   left |= mixer;
  76.   right |= mixer;
  77.   }
  78. if (bits == 0x03 || bits == 0x04)
  79.   {
  80.   mix_write(0x80 | bits, 0x078B);
  81.   mix_write(left, 0x078B);
  82.   right_vol = left_vol;
  83. } else
  84.   {
  85.   mix_write(0x80 | 0x20 | bits, 0x078B);
  86.   mix_write(left, 0x078B);
  87.   mix_write(0x80 | 0x40 | bits, 0x078B);
  88.   mix_write(right, 0x078B);
  89.   }
  90. return (left_vol | (right_vol << 8));
  91. }
  92. static void
  93. set_mode(int new_mode)
  94. {
  95. mix_write(0x80 | 0x05, 0x078B);
  96. mix_write(new_mode, 0x078B);
  97. mode_control = new_mode;
  98. }
  99. static int
  100. pas_mixer_set(int whichDev, unsigned int level)
  101. {
  102. int             left, right, devmask, changed, i, mixer = 0;
  103. DEB(printk("static int pas_mixer_set(int whichDev = %d, unsigned int level = %X)n", whichDev, level));
  104. left = level & 0x7f;
  105. right = (level & 0x7f00) >> 8;
  106. if (whichDev < SOUND_MIXER_NRDEVICES) {
  107. if ((1 << whichDev) & rec_devices)
  108. mixer = 0x20;
  109. else
  110. mixer = 0x00;
  111. }
  112. switch (whichDev)
  113.   {
  114.   case SOUND_MIXER_VOLUME: /* Master volume (0-63) */
  115.   levels[whichDev] = mixer_output(right, left, 63, 0x01, 0);
  116.   break;
  117.   /*
  118.    * Note! Bass and Treble are mono devices. Will use just the left
  119.    * channel.
  120.    */
  121.   case SOUND_MIXER_BASS: /* Bass (0-12) */
  122.   levels[whichDev] = mixer_output(right, left, 12, 0x03, 0);
  123.   break;
  124.   case SOUND_MIXER_TREBLE: /* Treble (0-12) */
  125.   levels[whichDev] = mixer_output(right, left, 12, 0x04, 0);
  126.   break;
  127.   case SOUND_MIXER_SYNTH: /* Internal synthesizer (0-31) */
  128.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x00, mixer);
  129.   break;
  130.   case SOUND_MIXER_PCM: /* PAS PCM (0-31) */
  131.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x05, mixer);
  132.   break;
  133.   case SOUND_MIXER_ALTPCM: /* SB PCM (0-31) */
  134.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x07, mixer);
  135.   break;
  136.   case SOUND_MIXER_SPEAKER: /* PC speaker (0-31) */
  137.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x06, mixer);
  138.   break;
  139.   case SOUND_MIXER_LINE: /* External line (0-31) */
  140.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x02, mixer);
  141.   break;
  142.   case SOUND_MIXER_CD: /* CD (0-31) */
  143.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x03, mixer);
  144.   break;
  145.   case SOUND_MIXER_MIC: /* External microphone (0-31) */
  146.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x04, mixer);
  147.   break;
  148.   case SOUND_MIXER_IMIX: /* Recording monitor (0-31) (Output mixer only) */
  149.   levels[whichDev] = mixer_output(right, left, 31, 0x10 | 0x01,
  150.   0x00);
  151.   break;
  152.   case SOUND_MIXER_RECLEV: /* Recording level (0-15) */
  153.   levels[whichDev] = mixer_output(right, left, 15, 0x02, 0);
  154.   break;
  155.   case SOUND_MIXER_RECSRC:
  156.   devmask = level & POSSIBLE_RECORDING_DEVICES;
  157.   changed = devmask ^ rec_devices;
  158.   rec_devices = devmask;
  159.   for (i = 0; i < SOUND_MIXER_NRDEVICES; i++)
  160.   if (changed & (1 << i))
  161.     {
  162.     pas_mixer_set(i, levels[i]);
  163.     }
  164.   return rec_devices;
  165.   break;
  166.   default:
  167.   return -EINVAL;
  168.   }
  169. return (levels[whichDev]);
  170. }
  171. /*****/
  172. static void
  173. pas_mixer_reset(void)
  174. {
  175. int             foo;
  176. DEB(printk("pas2_mixer.c: void pas_mixer_reset(void)n"));
  177. for (foo = 0; foo < SOUND_MIXER_NRDEVICES; foo++)
  178. pas_mixer_set(foo, levels[foo]);
  179. set_mode(0x04 | 0x01);
  180. }
  181. static int pas_mixer_ioctl(int dev, unsigned int cmd, caddr_t arg)
  182. {
  183. int level,v ;
  184. DEB(printk("pas2_mixer.c: int pas_mixer_ioctl(unsigned int cmd = %X, unsigned int arg = %X)n", cmd, arg));
  185. if (cmd == SOUND_MIXER_PRIVATE1) { /* Set loudness bit */
  186. if (get_user(level, (int *)arg))
  187. return -EFAULT;
  188. if (level == -1)  /* Return current settings */
  189. level = (mode_control & 0x04);
  190. else {
  191. mode_control &= ~0x04;
  192. if (level)
  193. mode_control |= 0x04;
  194. set_mode(mode_control);
  195. }
  196. level = !!level;
  197. return put_user(level, (int *)arg);
  198. }
  199. if (cmd == SOUND_MIXER_PRIVATE2) { /* Set enhance bit */
  200. if (get_user(level, (int *)arg))
  201. return -EFAULT;
  202. if (level == -1) { /* Return current settings */
  203. if (!(mode_control & 0x03))
  204. level = 0;
  205. else
  206. level = ((mode_control & 0x03) + 1) * 20;
  207. } else {
  208. int i = 0;
  209. level &= 0x7f;
  210. if (level)
  211. i = (level / 20) - 1;
  212. mode_control &= ~0x03;
  213. mode_control |= i & 0x03;
  214. set_mode(mode_control);
  215. if (i)
  216. i = (i + 1) * 20;
  217. level = i;
  218. }
  219. return put_user(level, (int *)arg);
  220. }
  221. if (cmd == SOUND_MIXER_PRIVATE3) { /* Set mute bit */
  222. if (get_user(level, (int *)arg))
  223. return -EFAULT;
  224. if (level == -1) /* Return current settings */
  225. level = !(pas_read(0x0B8A) & 0x20);
  226. else {
  227. if (level)
  228. pas_write(pas_read(0x0B8A) & (~0x20), 0x0B8A);
  229. else
  230. pas_write(pas_read(0x0B8A) | 0x20, 0x0B8A);
  231. level = !(pas_read(0x0B8A) & 0x20);
  232. }
  233. return put_user(level, (int *)arg);
  234. }
  235. if (((cmd >> 8) & 0xff) == 'M') {
  236. if (get_user(v, (int *)arg))
  237. return -EFAULT;
  238. if (_SIOC_DIR(cmd) & _SIOC_WRITE) {
  239. v = pas_mixer_set(cmd & 0xff, v);
  240. } else {
  241. switch (cmd & 0xff) {
  242. case SOUND_MIXER_RECSRC:
  243. v = rec_devices;
  244. break;
  245. case SOUND_MIXER_STEREODEVS:
  246. v = SUPPORTED_MIXER_DEVICES & ~(SOUND_MASK_BASS | SOUND_MASK_TREBLE);
  247. break;
  248. case SOUND_MIXER_DEVMASK:
  249. v = SUPPORTED_MIXER_DEVICES;
  250. break;
  251. case SOUND_MIXER_RECMASK:
  252. v = POSSIBLE_RECORDING_DEVICES & SUPPORTED_MIXER_DEVICES;
  253. break;
  254. case SOUND_MIXER_CAPS:
  255. v = 0; /* No special capabilities */
  256. break;
  257. default:
  258. v = levels[cmd & 0xff];
  259. break;
  260. }
  261. }
  262. return put_user(v, (int *)arg);
  263. }
  264. return -EINVAL;
  265. }
  266. static struct mixer_operations pas_mixer_operations =
  267. {
  268. owner: THIS_MODULE,
  269. id: "PAS16",
  270. name: "Pro Audio Spectrum 16",
  271. ioctl: pas_mixer_ioctl
  272. };
  273. int __init
  274. pas_init_mixer(void)
  275. {
  276. int             d;
  277. levels = load_mixer_volumes("PAS16_1", default_levels, 1);
  278. pas_mixer_reset();
  279. if ((d = sound_alloc_mixerdev()) != -1)
  280.   {
  281.   audio_devs[pas_audiodev]->mixer_dev = d;
  282.   mixer_devs[d] = &pas_mixer_operations;
  283.   }
  284. return 1;
  285. }