How to create a device node from the init_module code of a Linux kernel module?

后端 未结 3 694
一整个雨季
一整个雨季 2020-11-30 17:41

I am writing a module for the linux kernel and I want to create some device nodes in the init function

int init_module(void)
{
    Major = register_chrdev(0,         


        
相关标签:
3条回答
  • 2020-11-30 18:13
    static int __init ofcd_init(void) /* Constructor */
    {
        printk(KERN_INFO "Welcome!");
        if (alloc_chrdev_region(&first, 0, 1, "char_dev") < 0)  //$cat /proc/devices
        {
            return -1;
        }
        if ((cl = class_create(THIS_MODULE, "chardrv")) == NULL)    //$ls /sys/class
        {
            unregister_chrdev_region(first, 1);
            return -1;
        }
        if (device_create(cl, NULL, first, NULL, "mynull") == NULL) //$ls /dev/
        {
            class_destroy(cl);
            unregister_chrdev_region(first, 1);
            return -1;
        }
        cdev_init(&c_dev, &fops);
        if (cdev_add(&c_dev, first, 1) == -1)
        {
            device_destroy(cl, first);
            class_destroy(cl);
            unregister_chrdev_region(first, 1);
            return -1;
        }
        return 0;
    }
    
    0 讨论(0)
  • 2020-11-30 18:23

    Minimal runnable example

    Minimized from other answers. GitHub upstream with test setup.

    #include <linux/cdev.h>
    #include <linux/device.h>
    #include <linux/fs.h> /* register_chrdev, unregister_chrdev */
    #include <linux/module.h>
    #include <linux/seq_file.h> /* seq_read, seq_lseek, single_release */
    
    #define NAME "lkmc_character_device_create"
    
    static int major = -1;
    static struct cdev mycdev;
    static struct class *myclass = NULL;
    
    static int show(struct seq_file *m, void *v)
    {
        seq_printf(m, "abcd");
        return 0;
    }
    
    static int open(struct inode *inode, struct file *file)
    {
        return single_open(file, show, NULL);
    }
    
    static const struct file_operations fops = {
        .llseek = seq_lseek,
        .open = open,
        .owner = THIS_MODULE,
        .read = seq_read,
        .release = single_release,
    };
    
    static void cleanup(int device_created)
    {
        if (device_created) {
            device_destroy(myclass, major);
            cdev_del(&mycdev);
        }
        if (myclass)
            class_destroy(myclass);
        if (major != -1)
            unregister_chrdev_region(major, 1);
    }
    
    static int myinit(void)
    {
        int device_created = 0;
    
        /* cat /proc/devices */
        if (alloc_chrdev_region(&major, 0, 1, NAME "_proc") < 0)
            goto error;
        /* ls /sys/class */
        if ((myclass = class_create(THIS_MODULE, NAME "_sys")) == NULL)
            goto error;
        /* ls /dev/ */
        if (device_create(myclass, NULL, major, NULL, NAME "_dev") == NULL)
            goto error;
        device_created = 1;
        cdev_init(&mycdev, &fops);
        if (cdev_add(&mycdev, major, 1) == -1)
            goto error;
        return 0;
    error:
        cleanup(device_created);
        return -1;
    }
    
    static void myexit(void)
    {
        cleanup(1);
    }
    
    module_init(myinit)
    module_exit(myexit)
    MODULE_LICENSE("GPL");
    
    0 讨论(0)
  • 2020-11-30 18:40

    To have more control over the device numbers and the device creation you could do the following steps (instead of register_chrdev()):

    1. Call alloc_chrdev_region() to get a major number and a range of minor numbers to work with.
    2. Create device class for your devices with class_create().
    3. For each device, call cdev_init() and cdev_add() to add the character device to the system.
    4. For each device, call device_create(). As a result, among other things, Udev will create device nodes for your devices. No need for mknod or the like. device_create() also allows you to control the names of the devices.

    There are probably many examples of this on the Net, one of them is here.

    0 讨论(0)
提交回复
热议问题