ac97.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. #include <linux/module.h>
  2. #include <linux/kernel.h>
  3. #include <linux/init.h>
  4. #include "ac97.h"
  5. /* Flag for mono controls. */
  6. #define MO 0
  7. /* And for stereo. */
  8. #define ST 1
  9. /* Whether or not the bits in the channel are inverted. */
  10. #define INV 1
  11. #define NINV 0
  12. static struct ac97_chn_desc {
  13. int ac97_regnum;
  14. int oss_channel;
  15. int maxval;
  16. int is_stereo;
  17. int oss_mask;
  18. int recordNum;
  19. u16 regmask;
  20. int is_inverted;
  21. } mixerRegs[] = {
  22. { AC97_MASTER_VOL_STEREO, SOUND_MIXER_VOLUME, 0x3f, ST, SOUND_MASK_VOLUME, 5, 0x0000, INV },
  23. { AC97_MASTER_VOL_MONO, SOUND_MIXER_PHONEOUT, 0x3f, MO, SOUND_MASK_PHONEOUT, 6, 0x0000, INV },
  24. { AC97_MASTER_TONE, SOUND_MIXER_TREBLE, 0x0f, MO, SOUND_MASK_TREBLE, -1, 0x00ff, INV },
  25. { AC97_MASTER_TONE, SOUND_MIXER_BASS, 0x0f, MO, SOUND_MASK_BASS, -1, 0xff00, INV },
  26. { AC97_PCBEEP_VOL, SOUND_MIXER_SPEAKER, 0x0f, MO, SOUND_MASK_SPEAKER, -1, 0x001e, INV },
  27. { AC97_PHONE_VOL, SOUND_MIXER_PHONEIN, 0x1f, MO, SOUND_MASK_PHONEIN, 7, 0x0000, INV },
  28. { AC97_MIC_VOL, SOUND_MIXER_MIC, 0x1f, MO, SOUND_MASK_MIC, 0, 0x0000, INV },
  29. { AC97_LINEIN_VOL, SOUND_MIXER_LINE, 0x1f, ST, SOUND_MASK_LINE, 4, 0x0000, INV },
  30. { AC97_CD_VOL, SOUND_MIXER_CD, 0x1f, ST, SOUND_MASK_CD, 1, 0x0000, INV },
  31. { AC97_VIDEO_VOL, SOUND_MIXER_VIDEO, 0x1f, ST, SOUND_MASK_VIDEO, 2, 0x0000, INV },
  32. { AC97_AUX_VOL, SOUND_MIXER_LINE1, 0x1f, ST, SOUND_MASK_LINE1, 3, 0x0000, INV },
  33. { AC97_PCMOUT_VOL, SOUND_MIXER_PCM, 0x1f, ST, SOUND_MASK_PCM, -1, 0x0000, INV },
  34. { AC97_RECORD_GAIN, SOUND_MIXER_IGAIN, 0x0f, ST, SOUND_MASK_IGAIN, -1, 0x0000, NINV },
  35. { -1, -1, 0xff, 0, 0, -1, 0x0000, 0 },
  36. };
  37. static struct ac97_chn_desc *
  38. ac97_find_chndesc (struct ac97_hwint *dev, int oss_channel)
  39. {
  40. int x;
  41. for (x = 0; mixerRegs[x].oss_channel != -1; x++) {
  42. if (mixerRegs[x].oss_channel == oss_channel)
  43. return mixerRegs + x;
  44. }
  45. return NULL;
  46. }
  47. static inline int
  48. ac97_is_valid_channel (struct ac97_hwint *dev, struct ac97_chn_desc *chn)
  49. {
  50. return (dev->last_written_mixer_values[chn->ac97_regnum / 2]
  51. != AC97_REG_UNSUPPORTED);
  52. }
  53. int
  54. ac97_init (struct ac97_hwint *dev)
  55. {
  56. int x;
  57. int reg0;
  58. /* Clear out the arrays of cached values. */
  59. for (x = 0; x < AC97_REG_CNT; x++)
  60. dev->last_written_mixer_values[x] = AC97_REGVAL_UNKNOWN;
  61. for (x = 0; x < SOUND_MIXER_NRDEVICES; x++)
  62. dev->last_written_OSS_values[x] = AC97_REGVAL_UNKNOWN;
  63. /* Clear the device masks. */
  64. dev->mixer_devmask = 0;
  65. dev->mixer_stereomask = 0;
  66. dev->mixer_recmask = 0;
  67. /* ??? Do a "standard reset" via register 0? */
  68. /* Hardware-dependent reset. */
  69. if (dev->reset_device (dev))
  70. return -1;
  71. /* Check the mixer device capabilities. */
  72. reg0 = dev->read_reg (dev, AC97_RESET);
  73. if (reg0 < 0)
  74. return -1;
  75. /* Check for support for treble/bass controls. */
  76. if (! (reg0 & 4)) {
  77. dev->last_written_mixer_values[AC97_MASTER_TONE / 2]
  78. = AC97_REG_UNSUPPORTED;
  79. }
  80. /* ??? There may be other tests here? */
  81. /* Fill in the device masks. */
  82. for (x = 0; mixerRegs[x].ac97_regnum != -1; x++) {
  83. if (ac97_is_valid_channel (dev, mixerRegs + x)) {
  84. dev->mixer_devmask |= mixerRegs[x].oss_mask;
  85. if (mixerRegs[x].is_stereo)
  86. dev->mixer_stereomask |= mixerRegs[x].oss_mask;
  87. if (mixerRegs[x].recordNum != -1)
  88. dev->mixer_recmask |= mixerRegs[x].oss_mask;
  89. }
  90. }
  91. return 0;
  92. }
  93. /* Reset the mixer to the currently saved settings. */
  94. int
  95. ac97_reset (struct ac97_hwint *dev)
  96. {
  97. int x;
  98. if (dev->reset_device (dev))
  99. return -1;
  100. /* Now set the registers back to their last-written values. */
  101. for (x = 0; mixerRegs[x].ac97_regnum != -1; x++) {
  102. int regnum = mixerRegs[x].ac97_regnum;
  103. int value = dev->last_written_mixer_values [regnum / 2];
  104. if (value >= 0)
  105. ac97_put_register (dev, regnum, value);
  106. }
  107. return 0;
  108. }
  109. /* Return the contents of register REG; use the cache if the value in it
  110. is valid. Returns a negative error code on failure. */
  111. static int
  112. ac97_get_register (struct ac97_hwint *dev, u8 reg)
  113. {
  114. if (reg > 127 || (reg & 1))
  115. return -EINVAL;
  116. /* See if it's in the cache, or if it's just plain invalid. */
  117. switch (dev->last_written_mixer_values[reg / 2]) {
  118. case AC97_REG_UNSUPPORTED:
  119. return -EINVAL;
  120. break;
  121. case AC97_REGVAL_UNKNOWN:
  122. dev->last_written_mixer_values[reg / 2] = dev->read_reg (dev, reg);
  123. break;
  124. default:
  125. break;
  126. }
  127. return dev->last_written_mixer_values[reg / 2];
  128. }
  129. /* Write VALUE to AC97 register REG, and cache its value in the last-written
  130. cache. Returns a negative error code on failure, or 0 on success. */
  131. int
  132. ac97_put_register (struct ac97_hwint *dev, u8 reg, u16 value)
  133. {
  134. if (reg > 127 || (reg & 1))
  135. return -EINVAL;
  136. if (dev->last_written_mixer_values[reg / 2] == AC97_REG_UNSUPPORTED)
  137. return -EINVAL;
  138. else {
  139. int res = dev->write_reg (dev, reg, value);
  140. if (res >= 0) {
  141. dev->last_written_mixer_values[reg / 2] = value;
  142. return 0;
  143. }
  144. else
  145. return res;
  146. }
  147. }
  148. /* Scale VALUE (a value fro 0 to MAXVAL) to a value from 0-100. If
  149. IS_STEREO is set, VALUE is a stereo value; the left channel value
  150. is in the lower 8 bits, and the right channel value is in the upper
  151. 8 bits.
  152. A negative error code is returned on failure, or the unsigned
  153. scaled value on success. */
  154. static int
  155. ac97_scale_to_oss_val (int value, int maxval, int is_stereo, int inv)
  156. {
  157. /* Muted? */
  158. if (value & AC97_MUTE)
  159. return 0;
  160. if (is_stereo)
  161. return (ac97_scale_to_oss_val (value & 255, maxval, 0, inv) << 8)
  162. | (ac97_scale_to_oss_val ((value >> 8) & 255, maxval, 0, inv) << 0);
  163. else {
  164. int i;
  165. /* Inverted. */
  166. if (inv)
  167. value = maxval - value;
  168. i = (value * 100 + (maxval / 2)) / maxval;
  169. if (i > 100)
  170. i = 100;
  171. if (i < 0)
  172. i = 0;
  173. return i;
  174. }
  175. }
  176. static int
  177. ac97_scale_from_oss_val (int value, int maxval, int is_stereo, int inv)
  178. {
  179. if (is_stereo)
  180. return (ac97_scale_from_oss_val (value & 255, maxval, 0, inv) << 8)
  181. | (ac97_scale_from_oss_val ((value >> 8) & 255, maxval, 0, inv) << 0);
  182. else {
  183. int i = ((value & 255) * maxval + 50) / 100;
  184. if (inv)
  185. i = maxval - i;
  186. if (i < 0)
  187. i = 0;
  188. if (i > maxval)
  189. i = maxval;
  190. return i;
  191. }
  192. }
  193. static int
  194. ac97_set_mixer (struct ac97_hwint *dev, int oss_channel, u16 oss_value)
  195. {
  196. int scaled_value;
  197. struct ac97_chn_desc *channel = ac97_find_chndesc (dev, oss_channel);
  198. int result;
  199. if (channel == NULL)
  200. return -ENODEV;
  201. if (! ac97_is_valid_channel (dev, channel))
  202. return -ENODEV;
  203. scaled_value = ac97_scale_from_oss_val (oss_value, channel->maxval,
  204. channel->is_stereo,
  205. channel->is_inverted);
  206. if (scaled_value < 0)
  207. return scaled_value;
  208. if (channel->regmask != 0) {
  209. int mv;
  210. int oldval = ac97_get_register (dev, channel->ac97_regnum);
  211. if (oldval < 0)
  212. return oldval;
  213. for (mv = channel->regmask; ! (mv & 1); mv >>= 1)
  214. scaled_value <<= 1;
  215. scaled_value &= channel->regmask;
  216. scaled_value |= (oldval & ~channel->regmask);
  217. }
  218. result = ac97_put_register (dev, channel->ac97_regnum, scaled_value);
  219. if (result == 0)
  220. dev->last_written_OSS_values[oss_channel] = oss_value;
  221. return result;
  222. }
  223. static int
  224. ac97_get_mixer_scaled (struct ac97_hwint *dev, int oss_channel)
  225. {
  226. struct ac97_chn_desc *channel = ac97_find_chndesc (dev, oss_channel);
  227. int regval;
  228. if (channel == NULL)
  229. return -ENODEV;
  230. if (! ac97_is_valid_channel (dev, channel))
  231. return -ENODEV;
  232. regval = ac97_get_register (dev, channel->ac97_regnum);
  233. if (regval < 0)
  234. return regval;
  235. if (channel->regmask != 0) {
  236. int mv;
  237. regval &= channel->regmask;
  238. for (mv = channel->regmask; ! (mv & 1); mv >>= 1)
  239. regval >>= 1;
  240. }
  241. return ac97_scale_to_oss_val (regval, channel->maxval,
  242. channel->is_stereo,
  243. channel->is_inverted);
  244. }
  245. static int
  246. ac97_get_recmask (struct ac97_hwint *dev)
  247. {
  248. int recReg = ac97_get_register (dev, AC97_RECORD_SELECT);
  249. if (recReg < 0)
  250. return recReg;
  251. else {
  252. int x;
  253. for (x = 0; mixerRegs[x].ac97_regnum >= 0; x++) {
  254. if (mixerRegs[x].recordNum == (recReg & 7))
  255. return mixerRegs[x].oss_mask;
  256. }
  257. return -ENODEV;
  258. }
  259. }
  260. static int
  261. ac97_set_recmask (struct ac97_hwint *dev, int oss_recmask)
  262. {
  263. int x;
  264. if (oss_recmask == 0)
  265. oss_recmask = SOUND_MIXER_MIC;
  266. for (x = 0; mixerRegs[x].ac97_regnum >= 0; x++) {
  267. if ((mixerRegs[x].recordNum >= 0)
  268. && (oss_recmask & mixerRegs[x].oss_mask))
  269. break;
  270. }
  271. if (mixerRegs[x].ac97_regnum < 0)
  272. return -ENODEV;
  273. else {
  274. int regval = (mixerRegs[x].recordNum << 8) | mixerRegs[x].recordNum;
  275. int res = ac97_put_register (dev, AC97_RECORD_SELECT, regval);
  276. if (res == 0)
  277. return ac97_get_recmask (dev);
  278. else
  279. return res;
  280. }
  281. }
  282. /* Set the mixer DEV to the list of values in VALUE_LIST. Return 0 on
  283. success, or a negative error code. */
  284. int
  285. ac97_set_values (struct ac97_hwint *dev,
  286. struct ac97_mixer_value_list *value_list)
  287. {
  288. int x;
  289. for (x = 0; value_list[x].oss_channel != -1; x++) {
  290. int chnum = value_list[x].oss_channel;
  291. struct ac97_chn_desc *chent = ac97_find_chndesc (dev, chnum);
  292. if (chent != NULL) {
  293. u16 val;
  294. int res;
  295. if (chent->is_stereo)
  296. val = (value_list[x].value.stereo.right << 8)
  297. | value_list[x].value.stereo.left;
  298. else {
  299. /* We do this so the returned value looks OK in the
  300. mixer app. It's not necessary otherwise. */
  301. val = (value_list[x].value.mono << 8)
  302. | value_list[x].value.mono;
  303. }
  304. res = ac97_set_mixer (dev, chnum, val);
  305. if (res < 0)
  306. return res;
  307. }
  308. else
  309. return -ENODEV;
  310. }
  311. return 0;
  312. }
  313. int
  314. ac97_mixer_ioctl (struct ac97_hwint *dev, unsigned int cmd, void __user *arg)
  315. {
  316. int ret;
  317. switch (cmd) {
  318. case SOUND_MIXER_READ_RECSRC:
  319. ret = ac97_get_recmask (dev);
  320. break;
  321. case SOUND_MIXER_WRITE_RECSRC:
  322. {
  323. if (get_user (ret, (int __user *) arg))
  324. ret = -EFAULT;
  325. else
  326. ret = ac97_set_recmask (dev, ret);
  327. }
  328. break;
  329. case SOUND_MIXER_READ_CAPS:
  330. ret = SOUND_CAP_EXCL_INPUT;
  331. break;
  332. case SOUND_MIXER_READ_DEVMASK:
  333. ret = dev->mixer_devmask;
  334. break;
  335. case SOUND_MIXER_READ_RECMASK:
  336. ret = dev->mixer_recmask;
  337. break;
  338. case SOUND_MIXER_READ_STEREODEVS:
  339. ret = dev->mixer_stereomask;
  340. break;
  341. default:
  342. /* Read or write request. */
  343. ret = -EINVAL;
  344. if (_IOC_TYPE (cmd) == 'M') {
  345. int dir = _SIOC_DIR (cmd);
  346. int channel = _IOC_NR (cmd);
  347. if (channel >= 0 && channel < SOUND_MIXER_NRDEVICES) {
  348. ret = 0;
  349. if (dir & _SIOC_WRITE) {
  350. int val;
  351. if (get_user (val, (int __user *) arg) == 0)
  352. ret = ac97_set_mixer (dev, channel, val);
  353. else
  354. ret = -EFAULT;
  355. }
  356. if (ret >= 0 && (dir & _SIOC_READ)) {
  357. if (dev->last_written_OSS_values[channel]
  358. == AC97_REGVAL_UNKNOWN)
  359. dev->last_written_OSS_values[channel]
  360. = ac97_get_mixer_scaled (dev, channel);
  361. ret = dev->last_written_OSS_values[channel];
  362. }
  363. }
  364. }
  365. break;
  366. }
  367. if (ret < 0)
  368. return ret;
  369. else
  370. return put_user(ret, (int __user *) arg);
  371. }
  372. EXPORT_SYMBOL(ac97_init);
  373. EXPORT_SYMBOL(ac97_set_values);
  374. EXPORT_SYMBOL(ac97_put_register);
  375. EXPORT_SYMBOL(ac97_mixer_ioctl);
  376. EXPORT_SYMBOL(ac97_reset);
  377. MODULE_LICENSE("GPL");
  378. /*
  379. * Local variables:
  380. * c-basic-offset: 4
  381. * End:
  382. */