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

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.  * Video capture interface for Linux
  3.  *
  4.  * A generic video device interface for the LINUX operating system
  5.  * using a set of device structures/vectors for low level operations.
  6.  *
  7.  * This program is free software; you can redistribute it and/or
  8.  * modify it under the terms of the GNU General Public License
  9.  * as published by the Free Software Foundation; either version
  10.  * 2 of the License, or (at your option) any later version.
  11.  *
  12.  * Author: Alan Cox, <alan@redhat.com>
  13.  *
  14.  * Fixes: 20000516  Claudio Matsuoka <claudio@conectiva.com>
  15.  * - Added procfs support
  16.  */
  17. #include <linux/config.h>
  18. #include <linux/version.h>
  19. #include <linux/module.h>
  20. #include <linux/types.h>
  21. #include <linux/kernel.h>
  22. #include <linux/sched.h>
  23. #include <linux/smp_lock.h>
  24. #include <linux/mm.h>
  25. #include <linux/string.h>
  26. #include <linux/errno.h>
  27. #include <linux/videodev.h>
  28. #include <linux/init.h>
  29. #include <asm/uaccess.h>
  30. #include <asm/system.h>
  31. #include <asm/semaphore.h>
  32. #include <linux/kmod.h>
  33. #define VIDEO_NUM_DEVICES 256 
  34. /*
  35.  * Active devices 
  36.  */
  37.  
  38. static struct video_device *video_device[VIDEO_NUM_DEVICES];
  39. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  40. #include <linux/proc_fs.h>
  41. struct videodev_proc_data {
  42. struct list_head proc_list;
  43. char name[16];
  44. struct video_device *vdev;
  45. struct proc_dir_entry *proc_entry;
  46. };
  47. static struct proc_dir_entry *video_dev_proc_entry = NULL;
  48. struct proc_dir_entry *video_proc_entry = NULL;
  49. EXPORT_SYMBOL(video_proc_entry);
  50. LIST_HEAD(videodev_proc_list);
  51. #endif /* CONFIG_PROC_FS && CONFIG_VIDEO_PROC_FS */
  52. /*
  53.  * Read will do some smarts later on. Buffer pin etc.
  54.  */
  55.  
  56. static ssize_t video_read(struct file *file,
  57. char *buf, size_t count, loff_t *ppos)
  58. {
  59. struct video_device *vfl=video_device[MINOR(file->f_dentry->d_inode->i_rdev)];
  60. if(vfl->read)
  61. return vfl->read(vfl, buf, count, file->f_flags&O_NONBLOCK);
  62. else
  63. return -EINVAL;
  64. }
  65. /*
  66.  * Write for now does nothing. No reason it shouldnt do overlay setting
  67.  * for some boards I guess..
  68.  */
  69. static ssize_t video_write(struct file *file, const char *buf, 
  70. size_t count, loff_t *ppos)
  71. {
  72. struct video_device *vfl=video_device[MINOR(file->f_dentry->d_inode->i_rdev)];
  73. if(vfl->write)
  74. return vfl->write(vfl, buf, count, file->f_flags&O_NONBLOCK);
  75. else
  76. return 0;
  77. }
  78. /*
  79.  * Poll to see if we're readable, can probably be used for timing on incoming
  80.  * frames, etc..
  81.  */
  82. static unsigned int video_poll(struct file *file, poll_table * wait)
  83. {
  84. struct video_device *vfl=video_device[MINOR(file->f_dentry->d_inode->i_rdev)];
  85. if(vfl->poll)
  86. return vfl->poll(vfl, file, wait);
  87. else
  88. return 0;
  89. }
  90. /*
  91.  * Open a video device.
  92.  */
  93. static int video_open(struct inode *inode, struct file *file)
  94. {
  95. unsigned int minor = MINOR(inode->i_rdev);
  96. int err, retval = 0;
  97. struct video_device *vfl;
  98. if(minor>=VIDEO_NUM_DEVICES)
  99. return -ENODEV;
  100. lock_kernel();
  101. vfl=video_device[minor];
  102. if(vfl==NULL) {
  103. char modname[20];
  104. sprintf (modname, "char-major-%d-%d", VIDEO_MAJOR, minor);
  105. request_module(modname);
  106. vfl=video_device[minor];
  107. if (vfl==NULL) {
  108. retval = -ENODEV;
  109. goto error_out;
  110. }
  111. }
  112. if(vfl->busy) {
  113. retval = -EBUSY;
  114. goto error_out;
  115. }
  116. vfl->busy=1; /* In case vfl->open sleeps */
  117. if(vfl->owner)
  118. __MOD_INC_USE_COUNT(vfl->owner);
  119. if(vfl->open)
  120. {
  121. err=vfl->open(vfl,0); /* Tell the device it is open */
  122. if(err)
  123. {
  124. vfl->busy=0;
  125. if(vfl->owner)
  126. __MOD_DEC_USE_COUNT(vfl->owner);
  127. unlock_kernel();
  128. return err;
  129. }
  130. }
  131. unlock_kernel();
  132. return 0;
  133. error_out:
  134. unlock_kernel();
  135. return retval;
  136. }
  137. /*
  138.  * Last close of a video for Linux device
  139.  */
  140. static int video_release(struct inode *inode, struct file *file)
  141. {
  142. struct video_device *vfl;
  143. lock_kernel();
  144. vfl=video_device[MINOR(inode->i_rdev)];
  145. if(vfl->close)
  146. vfl->close(vfl);
  147. vfl->busy=0;
  148. if(vfl->owner)
  149. __MOD_DEC_USE_COUNT(vfl->owner);
  150. unlock_kernel();
  151. return 0;
  152. }
  153. static int video_ioctl(struct inode *inode, struct file *file,
  154. unsigned int cmd, unsigned long arg)
  155. {
  156. struct video_device *vfl=video_device[MINOR(inode->i_rdev)];
  157. int err=vfl->ioctl(vfl, cmd, (void *)arg);
  158. if(err!=-ENOIOCTLCMD)
  159. return err;
  160. switch(cmd)
  161. {
  162. default:
  163. return -EINVAL;
  164. }
  165. }
  166. /*
  167.  * We need to do MMAP support
  168.  */
  169.   
  170. int video_mmap(struct file *file, struct vm_area_struct *vma)
  171. {
  172. int ret = -EINVAL;
  173. struct video_device *vfl=video_device[MINOR(file->f_dentry->d_inode->i_rdev)];
  174. if(vfl->mmap) {
  175. lock_kernel();
  176. ret = vfl->mmap(vfl, (char *)vma->vm_start, 
  177. (unsigned long)(vma->vm_end-vma->vm_start));
  178. unlock_kernel();
  179. }
  180. return ret;
  181. }
  182. /*
  183.  * /proc support
  184.  */
  185. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  186. /* Hmm... i'd like to see video_capability information here, but
  187.  * how can I access it (without changing the other drivers? -claudio
  188.  */
  189. static int videodev_proc_read(char *page, char **start, off_t off,
  190.        int count, int *eof, void *data)
  191. {
  192. char *out = page;
  193. struct video_device *vfd = data;
  194. struct videodev_proc_data *d;
  195. struct list_head *tmp;
  196. int len;
  197. char c = ' ';
  198. list_for_each (tmp, &videodev_proc_list) {
  199. d = list_entry(tmp, struct videodev_proc_data, proc_list);
  200. if (vfd == d->vdev)
  201. break;
  202. }
  203. /* Sanity check */
  204. if (tmp == &videodev_proc_list)
  205. goto skip;
  206. #define PRINT_VID_TYPE(x) do { if (vfd->type & x) 
  207. out += sprintf (out, "%c%s", c, #x); c='|';} while (0)
  208. out += sprintf (out, "name            : %sn", vfd->name);
  209. out += sprintf (out, "type            :");
  210. PRINT_VID_TYPE(VID_TYPE_CAPTURE);
  211. PRINT_VID_TYPE(VID_TYPE_TUNER);
  212. PRINT_VID_TYPE(VID_TYPE_TELETEXT);
  213. PRINT_VID_TYPE(VID_TYPE_OVERLAY);
  214. PRINT_VID_TYPE(VID_TYPE_CHROMAKEY);
  215. PRINT_VID_TYPE(VID_TYPE_CLIPPING);
  216. PRINT_VID_TYPE(VID_TYPE_FRAMERAM);
  217. PRINT_VID_TYPE(VID_TYPE_SCALES);
  218. PRINT_VID_TYPE(VID_TYPE_MONOCHROME);
  219. PRINT_VID_TYPE(VID_TYPE_SUBCAPTURE);
  220. PRINT_VID_TYPE(VID_TYPE_MPEG_DECODER);
  221. PRINT_VID_TYPE(VID_TYPE_MPEG_ENCODER);
  222. PRINT_VID_TYPE(VID_TYPE_MJPEG_DECODER);
  223. PRINT_VID_TYPE(VID_TYPE_MJPEG_ENCODER);
  224. out += sprintf (out, "n");
  225. out += sprintf (out, "hardware        : 0x%xn", vfd->hardware);
  226. #if 0
  227. out += sprintf (out, "channels        : %dn", d->vcap.channels);
  228. out += sprintf (out, "audios          : %dn", d->vcap.audios);
  229. out += sprintf (out, "maxwidth        : %dn", d->vcap.maxwidth);
  230. out += sprintf (out, "maxheight       : %dn", d->vcap.maxheight);
  231. out += sprintf (out, "minwidth        : %dn", d->vcap.minwidth);
  232. out += sprintf (out, "minheight       : %dn", d->vcap.minheight);
  233. #endif
  234. skip:
  235. len = out - page;
  236. len -= off;
  237. if (len < count) {
  238. *eof = 1;
  239. if (len <= 0)
  240. return 0;
  241. } else
  242. len = count;
  243. *start = page + off;
  244. return len;
  245. }
  246. static void videodev_proc_create(void)
  247. {
  248. video_proc_entry = create_proc_entry("video", S_IFDIR, &proc_root);
  249. if (video_proc_entry == NULL) {
  250. printk("video_dev: unable to initialise /proc/videon");
  251. return;
  252. }
  253. video_proc_entry->owner = THIS_MODULE;
  254. video_dev_proc_entry = create_proc_entry("dev", S_IFDIR, video_proc_entry);
  255. if (video_dev_proc_entry == NULL) {
  256. printk("video_dev: unable to initialise /proc/video/devn");
  257. return;
  258. }
  259. video_dev_proc_entry->owner = THIS_MODULE;
  260. }
  261. #ifdef MODULE
  262. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  263. static void videodev_proc_destroy(void)
  264. {
  265. if (video_dev_proc_entry != NULL)
  266. remove_proc_entry("dev", video_proc_entry);
  267. if (video_proc_entry != NULL)
  268. remove_proc_entry("video", &proc_root);
  269. }
  270. #endif
  271. #endif
  272. static void videodev_proc_create_dev (struct video_device *vfd, char *name)
  273. {
  274. struct videodev_proc_data *d;
  275. struct proc_dir_entry *p;
  276. if (video_dev_proc_entry == NULL)
  277. return;
  278. d = kmalloc (sizeof (struct videodev_proc_data), GFP_KERNEL);
  279. if (!d)
  280. return;
  281. p = create_proc_entry(name, S_IFREG|S_IRUGO|S_IWUSR, video_dev_proc_entry);
  282. if (!p)
  283. return;
  284. p->data = vfd;
  285. p->read_proc = videodev_proc_read;
  286. d->proc_entry = p;
  287. d->vdev = vfd;
  288. strcpy (d->name, name);
  289. /* How can I get capability information ? */
  290. list_add (&d->proc_list, &videodev_proc_list);
  291. }
  292. static void videodev_proc_destroy_dev (struct video_device *vfd)
  293. {
  294. struct list_head *tmp;
  295. struct videodev_proc_data *d;
  296. list_for_each (tmp, &videodev_proc_list) {
  297. d = list_entry(tmp, struct videodev_proc_data, proc_list);
  298. if (vfd == d->vdev) {
  299. remove_proc_entry(d->name, video_dev_proc_entry);
  300. list_del (&d->proc_list);
  301. kfree(d);
  302. break;
  303. }
  304. }
  305. }
  306. #endif /* CONFIG_VIDEO_PROC_FS */
  307. extern struct file_operations video_fops;
  308. /**
  309.  * video_register_device - register video4linux devices
  310.  * @vfd:  video device structure we want to register
  311.  * @type: type of device to register
  312.  * @nr:   which device number (0 == /dev/video0, 1 == /dev/video1, ...
  313.  *             -1 == first free)
  314.  *
  315.  * The registration code assigns minor numbers based on the type
  316.  * requested. -ENFILE is returned in all the device slots for this
  317.  * category are full. If not then the minor field is set and the
  318.  * driver initialize function is called (if non %NULL).
  319.  *
  320.  * Zero is returned on success.
  321.  *
  322.  * Valid types are
  323.  *
  324.  * %VFL_TYPE_GRABBER - A frame grabber
  325.  *
  326.  * %VFL_TYPE_VTX - A teletext device
  327.  *
  328.  * %VFL_TYPE_VBI - Vertical blank data (undecoded)
  329.  *
  330.  * %VFL_TYPE_RADIO - A radio card
  331.  */
  332. static DECLARE_MUTEX(videodev_register_lock);
  333. int video_register_device(struct video_device *vfd, int type, int nr)
  334. {
  335. int i=0;
  336. int base;
  337. int err;
  338. int end;
  339. char *name_base;
  340. char name[16];
  341. switch(type)
  342. {
  343. case VFL_TYPE_GRABBER:
  344. base=0;
  345. end=64;
  346. name_base = "video";
  347. break;
  348. case VFL_TYPE_VTX:
  349. base=192;
  350. end=224;
  351. name_base = "vtx";
  352. break;
  353. case VFL_TYPE_VBI:
  354. base=224;
  355. end=240;
  356. name_base = "vbi";
  357. break;
  358. case VFL_TYPE_RADIO:
  359. base=64;
  360. end=128;
  361. name_base = "radio";
  362. break;
  363. default:
  364. return -1;
  365. }
  366. /* pick a minor number */
  367. down(&videodev_register_lock);
  368. if (-1 == nr) {
  369. /* use first free */
  370. for(i=base;i<end;i++)
  371. if (NULL == video_device[i])
  372. break;
  373. if (i == end) {
  374. up(&videodev_register_lock);
  375. return -ENFILE;
  376. }
  377. } else {
  378. /* use the one the driver asked for */
  379. i = base+nr;
  380. if (NULL != video_device[i]) {
  381. up(&videodev_register_lock);
  382. return -ENFILE;
  383. }
  384. }
  385. video_device[i]=vfd;
  386. vfd->minor=i;
  387. up(&videodev_register_lock);
  388. /* The init call may sleep so we book the slot out
  389.    then call */
  390. MOD_INC_USE_COUNT;
  391. if(vfd->initialize) {
  392. err=vfd->initialize(vfd);
  393. if(err<0) {
  394. video_device[i]=NULL;
  395. MOD_DEC_USE_COUNT;
  396. return err;
  397. }
  398. }
  399. sprintf (name, "v4l/%s%d", name_base, i - base);
  400. /*
  401.  * Start the device root only. Anything else
  402.  * has serious privacy issues.
  403.  */
  404. vfd->devfs_handle =
  405. devfs_register (NULL, name, DEVFS_FL_DEFAULT,
  406. VIDEO_MAJOR, vfd->minor,
  407. S_IFCHR | S_IRUSR | S_IWUSR,
  408. &video_fops,
  409. NULL);
  410. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  411. sprintf (name, "%s%d", name_base, i - base);
  412. videodev_proc_create_dev (vfd, name);
  413. #endif
  414. return 0;
  415. }
  416. /**
  417.  * video_unregister_device - unregister a video4linux device
  418.  * @vfd: the device to unregister
  419.  *
  420.  * This unregisters the passed device and deassigns the minor
  421.  * number. Future open calls will be met with errors.
  422.  */
  423.  
  424. void video_unregister_device(struct video_device *vfd)
  425. {
  426. if(video_device[vfd->minor]!=vfd)
  427. panic("vfd: bad unregister");
  428. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  429. videodev_proc_destroy_dev (vfd);
  430. #endif
  431. devfs_unregister (vfd->devfs_handle);
  432. video_device[vfd->minor]=NULL;
  433. MOD_DEC_USE_COUNT;
  434. }
  435. static struct file_operations video_fops=
  436. {
  437. owner: THIS_MODULE,
  438. llseek: no_llseek,
  439. read: video_read,
  440. write: video_write,
  441. ioctl: video_ioctl,
  442. mmap: video_mmap,
  443. open: video_open,
  444. release: video_release,
  445. poll: video_poll,
  446. };
  447. /*
  448.  * Initialise video for linux
  449.  */
  450.  
  451. static int __init videodev_init(void)
  452. {
  453. printk(KERN_INFO "Linux video capture interface: v1.00n");
  454. if(devfs_register_chrdev(VIDEO_MAJOR,"video_capture", &video_fops))
  455. {
  456. printk("video_dev: unable to get major %dn", VIDEO_MAJOR);
  457. return -EIO;
  458. }
  459. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  460. videodev_proc_create ();
  461. #endif
  462. return 0;
  463. }
  464. static void __exit videodev_exit(void)
  465. {
  466. #ifdef MODULE
  467. #if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
  468. videodev_proc_destroy ();
  469. #endif
  470. #endif
  471. devfs_unregister_chrdev(VIDEO_MAJOR, "video_capture");
  472. }
  473. module_init(videodev_init)
  474. module_exit(videodev_exit)
  475. EXPORT_SYMBOL(video_register_device);
  476. EXPORT_SYMBOL(video_unregister_device);
  477. MODULE_AUTHOR("Alan Cox");
  478. MODULE_DESCRIPTION("Device registrar for Video4Linux drivers");
  479. MODULE_LICENSE("GPL");
  480. /*
  481.  * Local variables:
  482.  * c-basic-offset: 8
  483.  * End:
  484.  */