亚洲av成人无遮挡网站在线观看,少妇性bbb搡bbb爽爽爽,亚洲av日韩精品久久久久久,兔费看少妇性l交大片免费,无码少妇一区二区三区

Chinaunix

標題: 利用udev在/dev下動態(tài)生成/移除設(shè)備文件 [打印本頁]

作者: albcamus    時間: 2007-03-21 09:32
標題: 利用udev在/dev下動態(tài)生成/移除設(shè)備文件
用udev在/dev/下動態(tài)生成設(shè)備文件,這樣用戶就不用手工調(diào)用mknod了。


利用的kernel API:
   
    class_create        :    創(chuàng)建class
    class_destroy        :    銷毀class
    class_device_create    :    創(chuàng)建device
    class_device_destroy    :    銷毀device


注意,這些API是2.6.13開始有的,在2.6.13之前,應(yīng)當使用

class_simple_create
class_simple_destroy
class_simple_device_add
class_simple_device_remove

這一系列,也就是ldd3第14章描述的。 詳見:
   
https://lwn.net/Articles/128644/

Output:
===========================================
[root@localhost dynamic_dev_node]# insmod ./dummy_dev.ko
[root@localhost dynamic_dev_node]# file /dev/dummy_dev0
/dev/dummy_dev0: character special (250/0)
[root@localhost dynamic_dev_node]# rmmod dummy_dev.ko
[root@localhost dynamic_dev_node]# file /dev/dummy_dev0
/dev/dummy_dev0: ERROR: cannot open `/dev/dummy_dev0' (No such file or directory)




  1. #include <linux/kernel.h>
  2. #include <linux/module.h>
  3. #include <linux/init.h>
  4. #include <linux/mm.h>
  5. #include <linux/fs.h>
  6. #include <linux/types.h>
  7. #include <linux/delay.h>
  8. #include <linux/moduleparam.h>
  9. #include <linux/slab.h>
  10. #include <linux/errno.h>
  11. #include <linux/ioctl.h>
  12. #include <linux/cdev.h>
  13. #include <linux/string.h>
  14. #include <linux/list.h>
  15. #include <linux/pci.h>
  16. #include <asm/uaccess.h>
  17. #include <asm/atomic.h>
  18. #include <asm/unistd.h>



  19. #define THIS_DESCRIPTION "\
  20. This module is a dummy device driver, it register\n\
  21. \t\ta char device, and utilize udev to create/destroy \n\
  22. \t\tdevice node under /dev/ dynamicallly."


  23. MODULE_LICENSE("GPL");
  24. MODULE_AUTHOR("albcamus <albcamus@gmail.com>");
  25. MODULE_DESCRIPTION(THIS_DESCRIPTION);



  26. #define DUMMY_MAJOR 250
  27. #define DUMMY_MINOR 0
  28. #define DUMMY_NAME "dummy_dev"





  29. /**
  30. * the open routine of 'dummy_dev'
  31. */
  32. static int dummy_open(struct inode *inode, struct file *file)
  33. {
  34.     printk("Open OK\n");
  35.     return 0;
  36. }

  37. /**
  38. * the write routine of 'dummy_dev'
  39. */
  40. static ssize_t dummy_write(struct file *filp, const char *bp, size_t count, loff_t *ppos)
  41. {
  42.     printk("Don't Write!\n");
  43.     return 0;
  44. }

  45. /**
  46. * the read routine of 'dummy_dev'
  47. */
  48. static ssize_t dummy_read(struct file *filp, char *bp, size_t count, loff_t *ppos)
  49. {
  50.     return 0;
  51. }

  52. /**
  53. * the ioctl routine of 'dummy_dev'
  54. */
  55. static int dummy_ioctl(struct inode *inode, struct file *filep,
  56.             unsigned int cmd, unsigned long arg)
  57. {

  58.     return 0;
  59. }


  60. /**
  61. * file_operations of 'dummy_dev'
  62. */
  63. static struct file_operations dummy_dev_ops = {
  64.     .owner = THIS_MODULE,
  65.     .open = dummy_open,
  66.     .read = dummy_read,
  67.     .write = dummy_write,
  68.     .ioctl = dummy_ioctl,
  69. };


  70. /**
  71. * struct cdev of 'dummy_dev'
  72. */
  73. struct cdev *my_cdev;
  74. struct class *my_class;



  75. static int __init my_init(void)
  76. {
  77.     int err, devno = MKDEV(DUMMY_MAJOR, DUMMY_MINOR);


  78.     /* register the 'dummy_dev' char device */
  79.     my_cdev = cdev_alloc();
  80.     cdev_init(my_cdev, &dummy_dev_ops);

  81.     my_cdev->owner = THIS_MODULE;

  82.     err = cdev_add(my_cdev, devno, 1);
  83.     if (err != 0)
  84.         printk("dummy pci device register failed!\n");

  85.     /* creating your own class */
  86.     my_class = class_create(THIS_MODULE, "dummy_class");
  87.     if(IS_ERR(my_class)) {
  88.         printk("Err: failed in creating class.\n");
  89.         return -1;
  90.     }

  91.     /* register your own device in sysfs, and this will cause udevd to create corresponding device node */
  92.     class_device_create(my_class, NULL, MKDEV(DUMMY_MAJOR, DUMMY_MINOR), NULL, DUMMY_NAME "%d", DUMMY_MINOR );



  93.     return 0;
  94. }

  95. static void __exit my_fini(void)
  96. {
  97.     printk("bye\n");

  98.     cdev_del(my_cdev);
  99.     //kfree(my_cdev); no use. because that cdev_del() will call kfree if neccessary.


  100.     class_device_destroy(my_class, MKDEV(DUMMY_MAJOR, DUMMY_MINOR));
  101.     class_destroy(my_class);

  102. }




  103. module_init(my_init);
  104. module_exit(my_fini);
復(fù)制代碼

作者: albcamus    時間: 2007-03-21 09:35
https://lwn.net/Articles/128644/

這里的文章很重要。  其他沒什么了,這么管理設(shè)備是要比mknod方便些。

另外請教, LINUX_KERNEL_VERSION宏是不是消失了啊, 怎么都不能用。 現(xiàn)在判斷內(nèi)核版本的方法是什么呢?
作者: albcamus    時間: 2007-03-21 09:46
原帖由 albcamus 于 2007-3-21 09:35 發(fā)表于 2樓  
https://lwn.net/Articles/128644/

這里的文章很重要。  其他沒什么了,這么管理設(shè)備是要比mknod方便些。

另外請教, LINUX_KERNEL_VERSION宏是不是消失了啊, 怎么都不能用。 現(xiàn)在判斷內(nèi)核版本的方法是 ...



找到了,是這個:
         if ( LINUX_VERSION_CODE > KERNEL_VERSION(2,4,9) )
作者: bekars    時間: 2007-03-22 09:51
謝謝,寫得真清楚,mknod確實很麻煩
作者: bitzilla    時間: 2007-04-19 20:35
以前的時候都是用register_char_dev()來注冊一個字符設(shè)備,現(xiàn)在都用struct cdev了,
剛開始不習慣呀。
作者: bitzilla    時間: 2007-04-19 21:35
insmod simple_dev.ko出錯



simple_dev: module license 'unspecified' taints kernel.
simple_dev: Unknown symbol class_destroy
simple_dev: Unknown symbol class_create
simple_dev: Unknown symbol class_device_create
simple_dev: Unknown symbol class_device_destroy
作者: albcamus    時間: 2007-04-20 08:08
標題: 回復(fù) #6 bitzilla 的帖子
uname -r的輸出?
作者: bitzilla    時間: 2007-04-20 10:13
[minjun@localhost ~]$ uname -r
2.6.18-8.el5xen

我從/proc/kallsyms中可以找到有這些symbols

[minjun@localhost ~]$ cat /proc/kallsyms |grep class_create
c0534a86 T sysdev_class_create_file
c0536081 t class_create_release
c05368b8 T class_create_file
c0536baf T class_create
c05368b8 U class_create_file    [drm]
c0536baf U class_create [drm]
c0536baf U class_create [blktap]
c05368b8 U class_create_file    [rfcomm]
c05368b8 U class_create_file    [l2cap]
c0536baf U class_create [bluetooth]
c0536baf U class_create [lp]
c0536baf U class_create [sg]
c0536baf U class_create [soundcore]

[ 本帖最后由 bitzilla 于 2007-4-20 10:14 編輯 ]
作者: albcamus    時間: 2007-04-20 10:20
標題: 回復(fù) #8 bitzilla 的帖子
這個Linux是XEN 的Dom0還是DomU?  代碼中有MODULE_LICENSE("GPL")怎么還會taint kernel呢? 奇怪了
作者: bitzilla    時間: 2007-04-20 16:17
Dom0吧,我沒裝虛擬機。

我抄你的程序的時候沒有全抄,把LICENSE省了
作者: albcamus    時間: 2007-04-20 16:49
原帖由 bitzilla 于 2007-4-20 16:17 發(fā)表于 10樓  
Dom0吧,我沒裝虛擬機。

我抄你的程序的時候沒有全抄,把LICENSE省了



那幾個符號是作為GPL symbol導(dǎo)出的:

EXPORT_SYMBOL_GPL(class_create_file)

你把LICENSE弄沒了, 當然kernel就不給你用gpl的符號了
作者: bitzilla    時間: 2007-04-20 22:01


:em12:

我試試。。!
作者: bitzilla    時間: 2007-04-20 22:03
這樣就沒有問題了,謝謝老大。!
作者: biglei    時間: 2007-04-24 15:16
我曾經(jīng)就是為了license的問題苦惱了好幾天,就是為了這幾個函數(shù)。當時用的是BSD的license的說。后來改回來了就好了。
作者: anhongkui    時間: 2008-02-15 11:05
我使用的是2.6.9內(nèi)核,所以想用class_simple接口生成/dev/myname設(shè)備,但是不知道為什么不行?

  1.        if ((my_class = class_simple_create(THIS_MODULE, MY_NAME)) == NULL)
  2.                 return -1;
  3.         class_simple_device_add(my_class, my_dev, NULL, MY_NAME);
復(fù)制代碼


mydev是上邊注冊的一個字符設(shè)備的dev_t結(jié)構(gòu),已經(jīng)成功,major=253,minor=0

但是/dev目錄下沒有MY_NAME設(shè)備, 望解惑!
謝謝
作者: albcamus    時間: 2008-02-15 12:44
原帖由 anhongkui 于 2008-2-15 11:05 發(fā)表
我使用的是2.6.9內(nèi)核,所以想用class_simple接口生成/dev/myname設(shè)備,但是不知道為什么不行?

       if ((my_class = class_simple_create(THIS_MODULE, MY_NAME)) == NULL)
                return -1;
...


不太清楚, 我沒在2.6.13之前的內(nèi)核上試驗過, 你參照ldd3的做法也不成嗎?
作者: anhongkui    時間: 2008-02-15 13:05
我理解的, class_simple_device_add只是根據(jù)class_simple生成一個class_device,然后加入到sysfs中,不知道在哪里創(chuàng)建設(shè)備節(jié)點
網(wǎng)上搜到使用devfs_mk_cdev可以創(chuàng)建成功,但是我使用也不管用
作者: anhongkui    時間: 2008-02-15 13:29
devfs_mk_cdev在2.6.9中什么都不做,只是單純的返回0
作者: albcamus    時間: 2008-02-15 13:32
標題: 回復(fù) #18 anhongkui 的帖子
devfs造就是deprecated了,現(xiàn)在推薦用udev
作者: anhongkui    時間: 2008-02-15 13:35
原帖由 albcamus 于 2008-2-15 13:32 發(fā)表
devfs造就是deprecated了,現(xiàn)在推薦用udev



可是,現(xiàn)在的問題是找不到在2.6.9里邊用udev的方法
作者: anhongkui    時間: 2008-02-15 13:37

  1.         my_class = class_simple_create(THIS_MODULE, MY_NAME));
  2.         if (IS_ERR(my_class))
  3.                 return PTR_ERR(my_class);
  4.         class_simple_device_add(my_class, my_dev, NULL, MY_NAME);
復(fù)制代碼


我在內(nèi)核里邊搜到了一些使用class_simple_device_add的地方,都是跟我上邊的差不多,
就是不知道他們能不能自己創(chuàng)建設(shè)備文件
作者: anhongkui    時間: 2008-02-16 09:57
我想,還是mknod算了
作者: springtty    時間: 2008-07-07 13:26
開始我使用mknod方式,還想辦法在內(nèi)核中sys_mknod,但是失敗了
樓主牛B,我已是這個方法就OK,感謝樓主:)
作者: anhongkui    時間: 2009-01-20 15:55
該接口在較新的內(nèi)核如2.6.26里已經(jīng)沒有了

現(xiàn)在使用什么api??
作者: albcamus    時間: 2009-01-20 16:06
如果接口變了,請參考一下內(nèi)核里的代碼。 恕不更新了。

例如arch/x86/kernel/cpuid.c
作者: anhongkui    時間: 2009-01-20 19:55
chardev沒有弄出來,使用misc設(shè)備就可以了
misc_register就可以了完成了register_chardev_region和mknod兩個操作,
并且隱藏了major和minor,至少我不關(guān)心這兩個值
作者: shuiyu123    時間: 2009-04-07 11:21
標題: 回復(fù) #1 albcamus 的帖子
經(jīng)典!
我先學(xué)會用法,再領(lǐng)悟機制原理!真是不錯的好例程
多謝 albcamus 。!
作者: pilgrim_kevin    時間: 2010-02-24 15:49
目前內(nèi)核接口里已經(jīng)沒有class_device_create了。
作者: q631951221    時間: 2011-12-14 20:08
研究一下,最近在看驅(qū)動!
作者: NalaGinrut    時間: 2012-05-16 22:46
2.6.27以后class_device_create/class_device_destory改成device_create/device_destory了,大家記得再加一條判斷語句。

作者: linuxfellow    時間: 2012-05-17 08:31
回復(fù) 30# NalaGinrut
溫故而知新 ,好帖要常翻翻

   




歡迎光臨 Chinaunix (http://www.72891.cn/) Powered by Discuz! X3.2