视窗
loading...
您现在的位置:首页 > IT认证 > Linux认证 >

Linux自动创建设备节点


在驱动用加入对udev的支持主要做的就是:在驱动初始化的代码里调用class_create(…)为该设备创建一个class,再为每个设备调用device_create(…)( 在2.6较早的内核中用class_device_create)创建对应的设备。
   
    内核中定义的struct class结构体,顾名思义,一个struct class结构体类型变量对应一个类,内核同时提供了class_create(…)函数,可以用它来创建一个类,这个类存放于sysfs下面,一旦创建好了这个类,再调用 device_create(…)函数来在/dev目录下创建相应的设备节点。这样,加载模块的时候,用户空间中的udev会自动响应 device_create(…)函数,去/sysfs下寻找对应的类从而创建设备节点。
   
    struct class定义在头文件include/linux/device.h中class_create(…)在/drivers/base/class.c中实现device_create(…)函数在/drivers/base/core.c中实现class_destroy(…),device_destroy(…)也在/drivers/base/core.c中实现调用过程类似如下:
   
    static struct class *spidev_class;
   
    /*-------------------------------------------------------------------------*/
   
    static int spidev_probe(struct spi_device *spi)
   
    {
   
    …
   
    dev = device_create(spidev_class, &spi->dev, spidev->devt,
   
    spidev, "spidev%d.%d",
   
    spi->master->bus_num, spi->chip_select);
   
    …
   
    }
   
    static int spidev_remove(struct spi_device *spi)
   
    {
   
    ……
   
    device_destroy(spidev_class, spidev->devt);
   
    ……
   
    return 0;
   
    }
   
    static struct spi_driver spidev_spi = {
   
    .driver = {
   
    .name =        "spidev",
   
    .owner =    THIS_MODULE,
   
    },
   
    .probe =    spidev_probe,
   
    .remove =    __devexit_p(spidev_remove),
   
    };
   
    /*-------------------------------------------------------------------------*/
   
    static int __init spidev_init(void)
   
    {
   
    …
   
    spidev_class = class_create(THIS_MODULE, "spidev");
   
    if (IS_ERR(spidev_class)) {
   
    unregister_chrdev(SPIDEV_MAJOR, spidev_spi.driver.name);
   
    return PTR_ERR(spidev_class);
   
    }
   
    …
   
    }
   
    module_init(spidev_init);
   
    static void __exit spidev_exit(void)
   
    {
   
    ……
   
    class_destroy(spidev_class);
   
    ……
   
    }
   
    module_exit(spidev_exit);
   
    MODULE_DESCRIPTION("User mode SPI device interface");
   
    MODULE_LICENSE("GPL");
   
    下面以一个简单字符设备驱动来展示如何使用这几个函数
   
    #include <linux/module.h>
   
    #include <linux/kernel.h>
   
    #include <linux/init.h>
   
    #include <linux/fs.h>
   
    #include <linux/cdev.h>
   
    #include <linux/device.h>
   
    int HELLO_MAJOR = 0;
   
    int HELLO_MINOR = 0;
   
    int NUMBER_OF_DEVICES = 2;
   
    struct class *my_class;
   
    struct cdev cdev; dev_t devno;
   
    struct file_operations hello_fops = {
   
    .owner = THIS_MODULE
   
    };
   
    static int __init hello_init (void)
   
    {
   
    int result;
   
    devno = MKDEV(HELLO_MAJOR, HELLO_MINOR);
   
    if (HELLO_MAJOR)
   
    result = register_chrdev_region(devno, 2, "memdev");
   
    else
   
    {
   
    result = alloc_chrdev_region(&devno, 0, 2, "memdev");
   
    HELLO_MAJOR = MAJOR(devno);
   
    }
   
    printk("MAJOR IS %d\n",HELLO_MAJOR);
   
    my_class = class_create(THIS_MODULE,"hello_char_class");  //类名为hello_char_class
   
    if(IS_ERR(my_class))
   
    {
   
    printk("Err: failed in creating class.\n");
   
    return -1;
   
    }
   
    device_create(my_class,NULL,devno,NULL,"memdev");      //设备名为memdev
   
    if (result<0)
   
    {
   
    printk (KERN_WARNING "hello: can't get major number %d\n", HELLO_MAJOR);
   
    return result;
   
    }
   
    cdev_init(&cdev, &hello_fops);
   
    cdev.owner = THIS_MODULE;
   
    cdev_add(&cdev, devno, NUMBER_OF_DEVICES);
   
    printk (KERN_INFO "Character driver Registered\n");
   
    return 0;
   
    }
   
    static void __exit hello_exit (void)
   
    {
   
    cdev_del (&cdev);
   
    device_destroy(my_class, devno);         //delete device node under /dev//必须先删除设备,再删除class类
   
    class_destroy(my_class);                 //delete class created by us
   
    unregister_chrdev_region (devno,NUMBER_OF_DEVICES);
   
    printk (KERN_INFO "char driver cleaned up\n");
   
    }
   
    module_init (hello_init);
   
    module_exit (hello_exit);
   
    MODULE_LICENSE ("GPL");
   
    这样,模块加载后,就能在/dev目录下找到memdev这个设备节点了。
   
    例2:内核中的drivers/i2c/i2c-dev.c
   
    在i2cdev_attach_adapter中调用device_create(i2c_dev_class, &adap->dev,
   
    MKDEV(I2C_MAJOR, adap->nr), NULL,
   
    "i2c-%d", adap->nr);
   
    这样在dev目录就产生i2c-0  或i2c-1节点
   
    接下来就是udev应用,udev是应用层的东西,udev需要内核sysfs和tmpfa的支持,sysfs为udev提供设备入口和uevent通道,tmpfs为udev设备文件提供存放空间
   
    udev的源码可以在去相关网站下载,然后就是对其在运行环境下的移植,指定交叉编译环境,修改Makefile下的CROSS_COMPILE,如为mipsel-linux-,DESTDIR=xxx,或直接make CROSS_COMPILE=mipsel-linux-,DESTDIR=xxx 并install把主要生成的udevd、udevstart拷贝rootfs下的/sbin/目录内,udev的配置文件udev.conf和rules.d下的rules文件拷贝到rootfs下的/etc/目录内,并在rootfs/etc/init.d/rcS中添加以下几行:
   
    echo "Starting udevd…"
   
    /sbin/udevd --daemon
   
    /sbin/udevstart
   
    (原rcS内容如下:
   
    # mount filesystems
   
    /bin/mount -t proc /proc /proc
   
    /bin/mount -t sysfs sysfs /sys
   
    /bin/mount -t tmpfs tmpfs /dev
   
    # create necessary devices
   
    /bin/mknod /dev/null c 1 3
   
    /bin/mkdir /dev/pts
   
    /bin/mount -t devpts devpts /dev/pts
   
    /bin/mknod /dev/audio c 14 4
   
    /bin/mknod /dev/ts c 10 16
   
    )
   
    这样当系统启动后,udevd和udevstart就会解析配置文件,并自动在/dev下创建设备节点文件

闂傚倷绶氬ḿ褍螞瀹€鍕;闁跨噦鎷�

闂傚倷绶氬ḿ褍螞瀹€鍕;闁跨噦鎷�

闂傚倷绶氬ḿ褍螞濞嗘挸绀夐柡鍥ュ灩閸屻劑鏌曢崼婵囧閻庢艾顭烽弻銊モ攽閸℃ê鐝旂紓浣瑰敾缂嶄線寮婚敓鐘茬妞ゆ劧瀵岄埀顒侇殜閺岀喖鎽庨崒姘ギ闂佽鍟崶褔鍞堕梺缁樻煥閹芥粎绮旈鍕厽闁绘ê寮堕幖鎰版煟鎺抽崝灞藉祫闂佺鎻粻鎴g箽闂備浇娉曢崳锕傚箯閿燂拷
闂傚倷绀侀幖顐︽偋濠婂牆绀堟繛鎴欏灪閸嬬喐銇勯弽顐沪闁绘帡绠栭弻鏇熺箾閸喖濮庨梺璇叉唉椤曆嗗絹闂佹悶鍎荤徊鑺ユ櫠閹绘崡褰掓偂鎼淬垹娈楅悗娈垮枛閻栧吋淇婇悜鑺ユ櫆闁告挆鍐帗婵犵數鍋涢悺銊у垝瀹€鈧懞閬嶆嚃閳哄嫬小婵炲鍘ч悺銊╁吹閸岀偞鐓涢柛鎰╁妼閳ь剝宕电划鏃囥亹閹烘挾鍙嗗┑鐐村灱濞呮洜鈧熬鎷�闂備浇顕уù鐑藉极閹间降鈧焦绻濋崑顖氱秺瀹曞崬螣鐠囧樊娼梻浣风串缁蹭粙寮甸鍕仼闁告鍊戞惔銊ョ倞闁挎繂鎳庨埛澶嬬箾鐎电ǹ甯堕柟铏耿閻涱喚鈧綆鍠栫粻锝嗙節婵犲倸顏柟鏋姂濮婃椽宕ㄦ繝鍐f嫻缂備礁顑嗙敮锟犲箚閸ャ劌顕遍柡澶嬪灥閸炪劑姊洪幖鐐插姌闁告柨绉归敐鐐哄冀椤儱閰e畷鍫曟晲閸涱厸鎷ゅ┑鐐差嚟閸樠囧箠濮椻偓楠炲棝宕橀鑲╊槹濡炪倖鍔戦崹娲儊閺冣偓缁绘繈濮€閿濆棛銆愰柣搴㈣壘閹芥粌危閹扮増鏅搁柨鐕傛嫹闂傚倷鑳堕幊鎾绘偤閵娾晛鍨傞柛鎰ゴ閸亪鏌熺€电ǹ小闁绘帒锕ら埞鎴︽偐閸欏鎮欏┑鐐叉噷閸ㄥ綊婀侀梺鎸庣箓濡盯鎯屾惔銊︾厓鐟滄粓宕滃▎鎾崇疇闁归偊鍏橀弸鏃堟煙鏉堥箖妾柛瀣儔閺屾盯顢曢妶鍛€剧紓鍌氱М閸嬫捇姊绘担鐟扮亰闁绘帪绠撴俊鐢告倷閻㈢數顦梺鍝勫暙閻楀棛鐥閺屾盯骞囬娑氥€婄紓浣插亾闁跨噦鎷�闂備浇顕х花鑲╁緤婵犳熬缍栧鑸靛姇閸ㄥ倹绻濇繝鍌氼伀妞も晠鏀遍妵鍕箳閹存繃娈茬紓浣风贰閸o絽顕i崼鏇熷€烽柛顭戝亝閻濐亜鈹戦悙鑼闁搞劌澧庣划姘舵焼瀹ュ懐鍔撮梺鍛婂壃閸涱垼妲繝鐢靛Х椤d粙鍩€椤掆偓閸熷潡鍩€椤掑倹鍤€闁挎洏鍨洪幏鍛村礈閹绘帗顓块梻浣呵归張顒傛崲瀹ュ鑸归柟缁㈠枟閻撴瑩鎮楀☉娆嬬細濠⒀屽墯缁绘繈鍩€椤掍焦濯撮柛锔诲幘閹虫繈姊洪崜鑼帥闁稿鎳愮槐鐐哄焵椤掑嫭鈷戦柣鐔稿閹界娀鏌eΔ鍐ㄤ户闁瑰箍鍨归~婵嬵敄閼恒儳浜欓梻浣告惈濞诧箓鎯岄鐐床闁糕剝绋掗悡娆愩亜閹炬鍟版禒鏉戭渻閵堝棙澶勯柛鎾跺枎閻g兘鏁愰崱娆戠槇闂佸憡鍔忛弲鈺佄i鐐粹拺闁奸€涘嵆閸濈儤鎱ㄦ繝鍌ょ吋闁哄苯鐗撻獮姗€顢欓懖鈺婂悈闂備胶鎳撻悺銊у垝瀹ュ洤鍨濋柨鏇炲€归悡銉︾箾閹寸倖鎴濓耿閻楀牏绡€闁逞屽墴楠炲秹顢欓幆褍瑙︽繝鐢靛仜濡霉妞嬪海鐜绘俊銈呭暟绾惧ジ鏌¢崒娑卞劌闁稿骸绻掗埀顒冾潐閹哥ǹ螞濠靛棛鏆︽慨妯垮煐閸嬫劗绱撴担璐細鐟滅増鍨垮娲箰鎼粹€虫灆闂佺懓鍤栭幏锟�闂備浇顕уù鐑藉极閹间礁绠犻柟鎹愬煐閺嗘粍銇勯幇鍓佺暠缂佲偓閸℃ḿ绠鹃柟瀵镐紳椤忓牆鏋侀柛顐f礃閸婄數鐥鐐村婵炲吋鍔欓弻娑㈠Ω閿斿墽鐓佺紓浣稿€圭敮锟犮€佸Δ浣瑰缂佸鏅濋锔解拺閻熸瑥瀚欢鑼磼缂佹ê鐏寸€殿噮鍋婇、娆撴偩瀹€濠冪カ婵犳鍠楅妵娑㈠磻閹剧粯鐓冪憸婊堝礈濞嗘挸纾归柛婵勫劤缁€濠囨倵閿濆骸鏋熼柛搴$Ч閺屾盯寮撮妸銉ョ瑢閻熸粎澧楃敮妤呮偂閳ь剙顪冮妶鍡楃瑨闁挎洩濡囩划鍫ユ晸閿燂拷闂傚倷娴囨竟鍫熴仈缁嬫娼栧┑鐘崇閻掗箖鏌熺紒銏犳灈婵☆偅锕㈤弻锝夋偄缁嬫妫嗙紒缁㈠幐閸嬫捇姊绘担鐟邦嚋缂佸甯掗悾婵嬪箹娴e摜锛涢梺鍝勭Р閸斿酣銆呴悜鑺ョ叆闁绘洖鍊圭€氾拷:webmaster@jscj.com闂傚倷绶氬ḿ褍螞瀹€鍕;闁瑰墽绮悡鐔搞亜椤愵偄骞樼紒浣哄厴閺岋綁鏁傜捄銊х厯闂佽桨绀佺粔褰掑极閹剧粯鏅搁柨鐕傛嫹4008816886

相关文章

无相关信息
更新时间2022-03-13 11:11:02【至顶部↑】
联系我们 | 邮件: webmaster@jscj.com | 客服热线电话:4008816886(QQ同号) |  濠电姷鏁搁崕鎴犵礊閳ь剚銇勯弴鍡楀閸欏繘鏌i幇顕呮毌闁稿鎹囬悰顕€宕归鍙ョ棯婵犵數濮崑鎾绘煕閵夋垵鑻▓顐㈩渻閵堝棙顥嗘い鏂匡功閹广垽鏁撻敓锟�

付款方式留言簿投诉中心网站纠错二维码手机版

客服电话: