Linux(二)LED驱动程序框架(总线设备驱动)

总线-设备-驱动

总线-设备-驱动 又称为 设备驱动模型。
在这里插入图片描述

一、 概念

总线(bus):负责管理挂载对应总线的设备以及驱动;
设备(device):挂载在某个总线的物理设备;
驱动(driver):与特定设备相关的软件,负责初始化该设备以及提供一些操作该设备的操作方式;
类(class):对于具有相同功能的设备,归结到一种类别,进行分类管理;

二、 工作原理

以下只说 总线-设备-驱动 模式下的操作

总线:

总线管理着两个链表:设备链表 和 驱动链表。

1、注册设备

当我们向内核注册一个设备时,便插入到总线的设备链表。

2、注册驱动

当我们向内核注册一个驱动时,便插入到总线的驱动链表。

3、进行匹配

在插入的同时,总线会执行一个 bus_type 结构体中的 match 方法对新插入的 设备/驱动 进行匹配。(例如以名字的方式匹配。方式有很多总,下面再详细分析。)

4、匹配调用probe函数

匹配成功后,会调用 驱动 device_driver 结构体中的 probe 方法。(通常在 probe 中获取设备资源。具体有开发人员决定。)

5、移除设备或驱动

在移除设备或驱动时,会调用 device_driver 结构体中的 remove 方法。

三、具体实现过程(代码分析)

1、注册设备

定义好要用到的硬件资源赋值到resource结构体进行注册,例如我的板子的用户LED为GROUP_PIN(5,3)。

static struct resource resources[] = {
        {
                .start = GROUP_PIN(5,3),
                .flags = IORESOURCE_IRQ,
                .name = "100ask_led_pin",
        },
        {
                .start = GROUP_PIN(5,8),
                .flags = IORESOURCE_IRQ,
                .name = "100ask_led_pin",
        },
};

把刚才注册好的设备资源放到platform_device进行二次注册。

static struct platform_device board_A_led_dev = {
        .name = "100ask_led",
        .num_resources = ARRAY_SIZE(resources),
        .resource = resources,
        .dev = {
                .release = led_dev_release,
         },
};

在注册设备入口函数进行设备注册,就是在总线的设备列表能找到刚才注册的设备。

static int __init led_dev_init(void)
{
    int err;
    
    err = platform_device_register(&board_A_led_dev);   
    
    return 0;
}

2、注册驱动

将实体函数chip_demo_gpio_probechip_demo_gpio_remove注册到platform_driver结构体

static struct platform_driver chip_demo_gpio_driver = {
    .probe      = chip_demo_gpio_probe,
    .remove     = chip_demo_gpio_remove,
    .driver     = {
        .name   = "100ask_led",
    },
};

通过匹配原则在驱动注册时进行设备驱动匹配,匹配成功后会调用刚才的probe函数,这个函数会将刚才匹配的节点进行注册( led_class_create_device(g_ledcnt)),probe 函数中根据 platform_device 的资源确定了引脚,probe 将g_ledpins 数组填充,将来设备层调用的资源g_ledpins[g_ledcnt]都是来自于这。
我们用platform_device结构体来指定设备信息时,platform_drive是直接从platform_device中拿资源的,如下platform_get_resource函数。

static int chip_demo_gpio_probe(struct platform_device *pdev)
{
    struct resource *res;
    int i = 0;

    while (1)
    {
        res = platform_get_resource(pdev, IORESOURCE_IRQ, i++);
        if (!res)
            break;
        
        g_ledpins[g_ledcnt] = res->start;
        led_class_create_device(g_ledcnt);
        g_ledcnt++;
    }
    return 0;
    
}

platform_drive从platform_device中拿资源时正式之前注册设备时的资源

struct resource *platform_get_resource(struct platform_device *dev,
                                   unsigned int type, unsigned int num)
{
	int i;
	for (i = 0; i < dev->num_resources; i++) {
		struct resource *r = &dev->resource[i];
		if (type == resource_type(r) && num-- == 0)
			return r;
	}
	return NULL;
}

3、驱动层

驱动层没有变化,注册app对应到的实体函数,写好入口函数。

static struct file_operations led_drv = {
	.owner	 = THIS_MODULE,
	.open    = led_drv_open,
	.read    = led_drv_read,
	.write   = led_drv_write,
	.release = led_drv_close,
};

/* 1. 把file_operations结构体告诉内核:注册驱动程序                                */
/* 2. 谁来注册驱动程序啊?得有一个入口函数:安装驱动程序时,就会去调用这个入口函数 */
static int __init led_init(void)
{
	int err;
	
	printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
	major = register_chrdev(0, "100ask_led", &led_drv);  /* /dev/led */


	led_class = class_create(THIS_MODULE, "100ask_led_class");
	err = PTR_ERR(led_class);
	if (IS_ERR(led_class)) {
		printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
		unregister_chrdev(major, "led");
		return -1;
	}
	
	return 0;
}

举个例子,驱动层实体函数led_drv_write 会被app层write调用,led_drv_write中则调用注册好的ctl函数

static ssize_t led_drv_write (struct file *file, const char __user *buf, size_t size, loff_t *offset)
{
	int err;
	char status;
	struct inode *inode = file_inode(file);
	int minor = iminor(inode);
	
	printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
	err = copy_from_user(&status, buf, 1);

	/* 根据次设备号和status控制LED */
	p_led_opr->ctl(minor, status);
	
	return 1;
}

板级的ctl函数源码如下,GROUP(g_ledpins[which])中用到的g_ledpins资源就是刚才设备驱动匹配时probe函数中的g_ledpins资源。

static int board_demo_led_ctl (int which, char status) /* 控制LED, which-哪个LED, status:1-亮,0-灭 */
{
    //printk("%s %s line %d, led %d, %s\n", __FILE__, __FUNCTION__, __LINE__, which, status ? "on" : "off");
    printk("set led %s: group %d, pin %d\n", status ? "on" : "off", GROUP(g_ledpins[which]), PIN(g_ledpins[which]));

    switch(GROUP(g_ledpins[which]))
    {
        case 0:
        {
            printk("set pin of group 0 ...\n");
            break;
        }
        case 1:
        {
            printk("set pin of group 1 ...\n");
            break;
        }
        case 2:
        {
            printk("set pin of group 2 ...\n");
            break;
        }
        case 3:
        {
            printk("set pin of group 3 ...\n");
            break;
        }
         case 4:
        {
            printk("set pin of group 2 ...\n");
            break;
        }
        case 5:
        {
            //板载对应的port
            printk("set pin of group 3 ...\n");
            break;
        }
    }

    return 0;
}

static struct led_operations board_demo_led_opr = {
    .init = board_demo_led_init,
    .ctl  = board_demo_led_ctl,
};

总线设备驱动,实现了设备和驱动的分离,匹配算法和设备树中的方法是一致的,这种模型虽然易于扩展,但是,冗余代码太多, 修改引脚时设备端的代码需要重新编译。更换引脚时, 上图中的 led_drv.c 基本不用改, 但是需要修改 板级的指定硬件资源的结构体(如下图),这样的话一个板子对应一个C文件,会使得Linux内核非常庞大,垃圾文件太多,所以还需要继续用设备树的方式进行改进。

static struct resource resources[] = {
        {
                .start = GROUP_PIN(5,3),
                .flags = IORESOURCE_IRQ,
                .name = "100ask_led_pin",
        },
        {
                .start = GROUP_PIN(5,8),
                .flags = IORESOURCE_IRQ,
                .name = "100ask_led_pin",
        },
};
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux CAN总线驱动框架Linux内核中的一种设备驱动框架,用于支持CAN(Controller Area Network)总线设备的通信。CAN总线是一种用于在各种嵌入式系统中传输数据的串行通信协议,主要应用在汽车电子系统等领域。 Linux CAN总线驱动框架的核心是CAN子系统,它负责提供对CAN总线设备的抽象和管理。CAN子系统包括了CAN核心模块和多个CAN控制器驱动模块。 CAN核心模块是整个CAN子系统的核心,它提供了CAN总线的抽象接口和通用功能,如接口的注册和注销、帧发送和接收、错误处理等。CAN核心模块还负责管理CAN控制器驱动模块,并向上层应用程序提供统一的API。 CAN控制器驱动模块是针对不同CAN控制器硬件的驱动程序,用于与硬件进行通信并提供设备特定的功能。每个CAN控制器驱动模块都实现了统一的接口,以便CAN核心模块对它们进行管理和调用。 使用Linux CAN总线驱动框架,可以方便地开发和管理CAN总线设备。开发人员只需编写特定硬件的CAN控制器驱动模块,并在CAN核心模块中注册该驱动模块即可。应用程序则可以通过CAN核心模块提供的API来访问和控制CAN总线设备。 总之,Linux CAN总线驱动框架Linux内核中的一种设备驱动框架,用于支持CAN总线设备的通信。它提供了CAN核心模块和多个CAN控制器驱动模块,通过统一的API和管理机制,方便地管理和控制CAN总线设备。 ### 回答2: Linux CAN总线驱动框架是为了支持控制器区域网络(Controller Area Network,CAN)总线的通信功能而设计的。CAN总线是一种高可靠性的串行总线,广泛应用于汽车、工业控制和嵌入式系统等领域。 Linux的CAN总线驱动框架主要包括以下几个部分: 1. CAN核心模块:该模块提供了CAN总线的基本功能,包括CAN设备的注册、管理和通信接口的定义等。它为CAN总线驱动提供了统一的接口层,使得不同厂家和型号的CAN控制器都能够通过相同的API进行访问和操作。 2. CAN总线驱动:CAN总线驱动是与具体硬件相关的模块,它负责与CAN控制器进行底层通信,并提供接口给上层模块使用。不同硬件厂商的CAN控制器可能有不同的通信协议和操作方式,因此需要为每种类型的CAN控制器编写相应的驱动程序。 3. CAN协议栈:CAN协议栈是在CAN总线驱动基础上实现的高层软件模块,用于处理CAN总线上的数据帧以及相关的协议和业务逻辑。它通常包括CAN帧封装和解封装、错误检测和恢复、网络管理和调度等功能。 4. CAN应用接口:为了方便用户开发CAN应用程序Linux提供了一系列的CAN应用接口,如socketCAN接口。通过这些接口,用户可以直接使用标准的套接字编程接口来发送和接收CAN消息,实现与其他系统组件和外部设备的数据交换。 总的来说,Linux的CAN总线驱动框架为CAN总线的应用开发提供了一套完整的软件解决方案,使得开发者能够方便地使用CAN总线进行通信。这个框架的设计与实现,大大简化了CAN总线应用的开发工作,提高了系统的可靠性和灵活性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值