tvmixer.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:9k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. #include <linux/module.h>
  2. #include <linux/kernel.h>
  3. #include <linux/sched.h>
  4. #include <linux/string.h>
  5. #include <linux/timer.h>
  6. #include <linux/delay.h>
  7. #include <linux/errno.h>
  8. #include <linux/slab.h>
  9. #include <linux/i2c.h>
  10. #include <linux/videodev.h>
  11. #include <linux/init.h>
  12. #include <linux/kdev_t.h>
  13. #include <asm/semaphore.h>
  14. #include <linux/sound.h>
  15. #include <linux/soundcard.h>
  16. #include <asm/uaccess.h>
  17. #include "audiochip.h"
  18. #include "id.h"
  19. #define DEV_MAX  4
  20. static int debug = 0;
  21. static int devnr = -1;
  22. MODULE_PARM(debug,"i");
  23. MODULE_PARM(devnr,"i");
  24. MODULE_AUTHOR("Gerd Knorr");
  25. MODULE_LICENSE("GPL");
  26. /* ----------------------------------------------------------------------- */
  27. struct TVMIXER {
  28. struct i2c_client *dev;
  29. int minor;
  30. int count;
  31. };
  32. static struct TVMIXER devices[DEV_MAX];
  33. static int tvmixer_adapters(struct i2c_adapter *adap);
  34. static int tvmixer_clients(struct i2c_client *client);
  35. /* ----------------------------------------------------------------------- */
  36. static int mix_to_v4l(int i)
  37. {
  38. int r;
  39. r = ((i & 0xff) * 65536 + 50) / 100;
  40. if (r > 65535) r = 65535;
  41. if (r <     0) r =     0;
  42. return r;
  43. }
  44. static int v4l_to_mix(int i)
  45. {
  46. int r;
  47. r = (i * 100 + 32768) / 65536;
  48. if (r > 100) r = 100;
  49. if (r <   0) r =   0;
  50. return r | (r << 8);
  51. }
  52. static int v4l_to_mix2(int l, int r)
  53. {
  54. r = (r * 100 + 32768) / 65536;
  55. if (r > 100) r = 100;
  56. if (r <   0) r =   0;
  57. l = (l * 100 + 32768) / 65536;
  58. if (l > 100) l = 100;
  59. if (l <   0) l =   0;
  60. return (r << 8) | l;
  61. }
  62. static int tvmixer_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg)
  63. {
  64. struct video_audio va;
  65. int left,right,ret,val = 0;
  66.         struct TVMIXER *mix = file->private_data;
  67. struct i2c_client *client = mix->dev;
  68. if (NULL == client)
  69. return -ENODEV;
  70.         if (cmd == SOUND_MIXER_INFO) {
  71.                 mixer_info info;
  72.                 strncpy(info.id, "tv card", sizeof(info.id));
  73.                 strncpy(info.name, client->name, sizeof(info.name));
  74.                 info.modify_counter = 42 /* FIXME */;
  75.                 if (copy_to_user((void *)arg, &info, sizeof(info)))
  76.                         return -EFAULT;
  77.                 return 0;
  78.         }
  79.         if (cmd == SOUND_OLD_MIXER_INFO) {
  80.                 _old_mixer_info info;
  81.                 strncpy(info.id, "tv card", sizeof(info.id));
  82.                 strncpy(info.name, client->name, sizeof(info.name));
  83.                 if (copy_to_user((void *)arg, &info, sizeof(info)))
  84.                         return -EFAULT;
  85.                 return 0;
  86.         }
  87.         if (cmd == OSS_GETVERSION)
  88.                 return put_user(SOUND_VERSION, (int *)arg);
  89. if (_SIOC_DIR(cmd) & _SIOC_WRITE)
  90. if (get_user(val, (int *)arg))
  91. return -EFAULT;
  92. /* read state */
  93. memset(&va,0,sizeof(va));
  94. client->driver->command(client,VIDIOCGAUDIO,&va);
  95. switch (cmd) {
  96. case MIXER_READ(SOUND_MIXER_RECMASK):
  97. case MIXER_READ(SOUND_MIXER_CAPS):
  98. case MIXER_READ(SOUND_MIXER_RECSRC):
  99. case MIXER_WRITE(SOUND_MIXER_RECSRC):
  100. ret = 0;
  101. break;
  102. case MIXER_READ(SOUND_MIXER_STEREODEVS):
  103. ret = SOUND_MASK_VOLUME;
  104. break;
  105. case MIXER_READ(SOUND_MIXER_DEVMASK):
  106. ret = SOUND_MASK_VOLUME;
  107. if (va.flags & VIDEO_AUDIO_BASS)
  108. ret |= SOUND_MASK_BASS;
  109. if (va.flags & VIDEO_AUDIO_TREBLE)
  110. ret |= SOUND_MASK_TREBLE;
  111. break;
  112. case MIXER_WRITE(SOUND_MIXER_VOLUME):
  113. left  = mix_to_v4l(val);
  114. right = mix_to_v4l(val >> 8);
  115. va.volume  = MAX(left,right);
  116. va.balance = (32768*MIN(left,right)) / (va.volume ? va.volume : 1);
  117. va.balance = (left<right) ? (65535-va.balance) : va.balance;
  118. client->driver->command(client,VIDIOCSAUDIO,&va);
  119. client->driver->command(client,VIDIOCGAUDIO,&va);
  120. /* fall throuth */
  121. case MIXER_READ(SOUND_MIXER_VOLUME):
  122. left  = (MIN(65536 - va.balance,32768) *
  123.  va.volume) / 32768;
  124. right = (MIN(va.balance,32768) *
  125.  va.volume) / 32768;
  126. ret = v4l_to_mix2(left,right);
  127. break;
  128. case MIXER_WRITE(SOUND_MIXER_BASS):
  129. va.bass = mix_to_v4l(val);
  130. client->driver->command(client,VIDIOCSAUDIO,&va);
  131. client->driver->command(client,VIDIOCGAUDIO,&va);
  132. /* fall throuth  */
  133. case MIXER_READ(SOUND_MIXER_BASS):
  134. ret = v4l_to_mix(va.bass);
  135. break;
  136. case MIXER_WRITE(SOUND_MIXER_TREBLE):
  137. va.treble = mix_to_v4l(val);
  138. client->driver->command(client,VIDIOCSAUDIO,&va);
  139. client->driver->command(client,VIDIOCGAUDIO,&va);
  140. /* fall throuth */
  141. case MIXER_READ(SOUND_MIXER_TREBLE):
  142. ret = v4l_to_mix(va.treble);
  143. break;
  144. default:
  145. return -EINVAL;
  146. }
  147. if (put_user(ret, (int *)arg))
  148. return -EFAULT;
  149. return 0;
  150. }
  151. static int tvmixer_open(struct inode *inode, struct file *file)
  152. {
  153.         int i, minor = minor(inode->i_rdev);
  154.         struct TVMIXER *mix = NULL;
  155. struct i2c_client *client = NULL;
  156. for (i = 0; i < DEV_MAX; i++) {
  157. if (devices[i].minor == minor) {
  158. mix = devices+i;
  159. client = mix->dev;
  160. break;
  161. }
  162. }
  163. if (NULL == client)
  164. return -ENODEV;
  165. /* lock bttv in memory while the mixer is in use  */
  166. file->private_data = mix;
  167. if (client->adapter->inc_use)
  168. client->adapter->inc_use(client->adapter);
  169.         return 0;
  170. }
  171. static int tvmixer_release(struct inode *inode, struct file *file)
  172. {
  173. struct TVMIXER *mix = file->private_data;
  174. struct i2c_client *client;
  175. client = mix->dev;
  176. if (NULL == client) {
  177. return -ENODEV;
  178. }
  179. if (client->adapter->dec_use)
  180. client->adapter->dec_use(client->adapter);
  181. return 0;
  182. }
  183. static struct i2c_driver driver = {
  184. name:            "tv card mixer driver",
  185.         id:              I2C_DRIVERID_TVMIXER,
  186. flags:           I2C_DF_DUMMY,
  187.         attach_adapter:  tvmixer_adapters,
  188.         detach_client:   tvmixer_clients,
  189. };
  190. static struct file_operations tvmixer_fops = {
  191. owner: THIS_MODULE,
  192. llseek:         no_llseek,
  193. ioctl:          tvmixer_ioctl,
  194. open:           tvmixer_open,
  195. release:        tvmixer_release,
  196. };
  197. /* ----------------------------------------------------------------------- */
  198. static int tvmixer_adapters(struct i2c_adapter *adap)
  199. {
  200. int i;
  201. if (debug)
  202. printk("tvmixer: adapter %sn",adap->name);
  203. for (i=0; i<I2C_CLIENT_MAX; i++) {
  204. if (!adap->clients[i])
  205. continue;
  206. tvmixer_clients(adap->clients[i]);
  207. }
  208. return 0;
  209. }
  210. static int tvmixer_clients(struct i2c_client *client)
  211. {
  212. struct video_audio va;
  213. int i,minor;
  214. /* TV card ??? */
  215. switch (client->adapter->id) {
  216. case I2C_ALGO_BIT | I2C_HW_B_BT848:
  217. case I2C_ALGO_BIT | I2C_HW_B_RIVA:
  218. /* ok, have a look ... */
  219. break;
  220. default:
  221. /* ignore that one */
  222. if (debug)
  223. printk("tvmixer: %s is not a tv cardn",
  224.        client->adapter->name);
  225. return -1;
  226. }
  227. printk("tvmixer: debug: %sn",client->name);
  228. /* unregister ?? */
  229. for (i = 0; i < DEV_MAX; i++) {
  230. if (devices[i].dev == client) {
  231. /* unregister */
  232. unregister_sound_mixer(devices[i].minor);
  233. devices[i].dev = NULL;
  234. devices[i].minor = -1;
  235. printk("tvmixer: %s unregistered (#1)n",client->name);
  236. return 0;
  237. }
  238. }
  239. /* look for a free slot */
  240. for (i = 0; i < DEV_MAX; i++)
  241. if (NULL == devices[i].dev)
  242. break;
  243. if (i == DEV_MAX) {
  244. printk(KERN_WARNING "tvmixer: DEV_MAX too smalln");
  245. return -1;
  246. }
  247. /* audio chip with mixer ??? */
  248. if (NULL == client->driver->command) {
  249. if (debug)
  250. printk("tvmixer: %s: driver->command is NULLn",
  251.        client->driver->name);
  252. return -1;
  253. }
  254. memset(&va,0,sizeof(va));
  255. if (0 != client->driver->command(client,VIDIOCGAUDIO,&va)) {
  256. if (debug)
  257. printk("tvmixer: %s: VIDIOCGAUDIO failedn",
  258.        client->name);
  259. return -1;
  260. }
  261. if (0 == (va.flags & VIDEO_AUDIO_VOLUME)) {
  262. if (debug)
  263. printk("tvmixer: %s: has no volume controln",
  264.        client->name);
  265. return -1;
  266. }
  267. /* everything is fine, register */
  268. if ((minor = register_sound_mixer(&tvmixer_fops,devnr)) < 0) {
  269. printk(KERN_ERR "tvmixer: cannot allocate mixer devicen");
  270. return -1;
  271. }
  272. devices[i].minor = minor;
  273. devices[i].count = 0;
  274. devices[i].dev   = client;
  275. printk("tvmixer: %s (%s) registered with minor %dn",
  276.        client->name,client->adapter->name,minor);
  277. return 0;
  278. }
  279. /* ----------------------------------------------------------------------- */
  280. static int tvmixer_init_module(void)
  281. {
  282. int i;
  283. for (i = 0; i < DEV_MAX; i++)
  284. devices[i].minor = -1;
  285. i2c_add_driver(&driver);
  286. return 0;
  287. }
  288. static void tvmixer_cleanup_module(void)
  289. {
  290. int i;
  291. i2c_del_driver(&driver);
  292. for (i = 0; i < DEV_MAX; i++) {
  293. if (devices[i].minor != -1) {
  294. unregister_sound_mixer(devices[i].minor);
  295. printk("tvmixer: %s unregistered (#2)n",
  296.        devices[i].dev->name);
  297. }
  298. }
  299. }
  300. module_init(tvmixer_init_module);
  301. module_exit(tvmixer_cleanup_module);
  302. /*
  303.  * Overrides for Emacs so that we follow Linus's tabbing style.
  304.  * ---------------------------------------------------------------------------
  305.  * Local variables:
  306.  * c-basic-offset: 8
  307.  * End:
  308.  */