字符型设备驱动

最简单的字符型设备驱动,仅支持读写和文件创建

#include <linux/module.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/kernel.h>
#include <linux/slab.h>

#define DEVICE_NAME "mem_device" // 设备名称
#define MEM_SIZE 4096 // 内存大小

static dev_t dev;
static struct cdev c_dev;
static struct class *cl;
static char *device_memory;

// 设置全局静态变量保存 major 和 minor
static int major = 0;
static int minor = 0;

static int mem_device_open(struct inode *inode, struct file *file)
{
    return 0;
}

static int mem_device_release(struct inode *inode, struct file *file)
{
    return 0;
}

static ssize_t mem_device_read(struct file *file, char __user *buf, size_t count, loff_t *pos)
{
    int bytes_to_read = count;
    if (bytes_to_read > (MEM_SIZE - *pos)) {
        bytes_to_read = (MEM_SIZE - *pos);
    }
    if (copy_to_user(buf, &device_memory[*pos], bytes_to_read) != 0) {
        return -EFAULT;
    }
    *pos += bytes_to_read;
    return bytes_to_read;
}

static ssize_t mem_device_write(struct file *file, const char __user *buf, size_t count, loff_t *pos)
{
    int bytes_to_write = count;
    if (bytes_to_write > (MEM_SIZE - *pos)) {
        bytes_to_write = (MEM_SIZE - *pos);
    }
    if (copy_from_user(&device_memory[*pos], buf, bytes_to_write) != 0) {
        return -EFAULT;
    }
    *pos += bytes_to_write;
    return bytes_to_write;
}

static const struct file_operations mem_device_fops = {
    .owner = THIS_MODULE,
    .open = mem_device_open,
    .release = mem_device_release,
    .read = mem_device_read,
    .write = mem_device_write,
};

static int __init mem_driver_init(void)
{
    if (major != 0) {
        // 如果 major 不为 0,则使用指定的 major 和 minor
        dev = MKDEV(major, minor);
        if (register_chrdev_region(dev, 1, DEVICE_NAME) < 0) {
            // 如果注册失败,则动态分配设备号
            if (alloc_chrdev_region(&dev, 0, 1, DEVICE_NAME) < 0) {
                return -1;
            }
        }
    } else {
        // 如果 major 为 0,则动态分配设备号
        if (alloc_chrdev_region(&dev, 0, 1, DEVICE_NAME) < 0) {
            return -1;
        }
        // 保存动态分配的 major 和 minor
        major = MAJOR(dev);
        minor = MINOR(dev);
    }

    if ((cl = class_create(THIS_MODULE, DEVICE_NAME)) == NULL) {
        unregister_chrdev_region(dev, 1);
        return -1;
    }

    if (device_create(cl, NULL, dev, NULL, DEVICE_NAME) == NULL) {
        class_destroy(cl);
        unregister_chrdev_region(dev, 1);
        return -1;
    }

    cdev_init(&c_dev, &mem_device_fops);
    if (cdev_add(&c_dev, dev, 1) == -1) {
        device_destroy(cl, dev);
        class_destroy(cl);
        unregister_chrdev_region(dev, 1);
        return -1;
    }

    device_memory = kmalloc(MEM_SIZE, GFP_KERNEL);
    if (!device_memory) {
        cdev_del(&c_dev);
        device_destroy(cl, dev);
        class_destroy(cl);
        unregister_chrdev_region(dev, 1);
        return -1;
    }

    printk(KERN_INFO "Memory device driver loaded\n");
    return 0;
}

static void __exit mem_driver_exit(void)
{
    kfree(device_memory);
    cdev_del(&c_dev);
    device_destroy(cl, dev);
    class_destroy(cl);
    unregister_chrdev_region(dev, 1);
    printk(KERN_INFO "Memory device driver unloaded\n");
}

module_param(major, int, 0644); // 允许在加载模块时传递 major 参数
MODULE_PARM_DESC(major, "Major number");
module_param(minor, int, 0644); // 允许在加载模块时传递 minor 参数
MODULE_PARM_DESC(minor, "Minor number");

module_init(mem_driver_init);
module_exit(mem_driver_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Maxwell");
MODULE_DESCRIPTION("A simple character device driver for memory access");
  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值