LED子系统剖析

写之前,先看一张图:


上次说了LED驱动程序,Linux自身也携带了LED驱动,且是脱离平台的,即LED子系统。操作起来十分简单。但是它的实质却不是那么容易,研究了一个晚上,终于明白了其中一个文件的功能啦,机led-class.c文件。现在分享一下:

其实LED的驱动位于内核driver/leds目录下。核心文件有:led-class.c leds-s3c24xx.cleds-gpio.c 。先看其中一个文件 led-core.c文件。

一看就知道和类class脱不了关系。class有何作用呢?首先何所谓类呢?就是一组设备具有共同性而抽象出来的。其实leds-gpio.c和leds-s3c24xx.c功能都是差不多,两者是并列的,他们都共有的功能是在class里面实现的。在led-class.c文件实现的功能总的来说就是先建立一个类leds,然后在该类下建立一个设备节点,最后就在该设备节点下载建立几个属性文件。而建立类的交给函数leds_init(void)来完成,而在该类下建立设备节点,以及在该节点下建立属性文件,并对属性文件实现读写操作。

现在我们先看看第一个,就是init加载文件,第一句也是核心句,就是建立一个类leds,并且函数返回值

赋值给了led_cdev->dev,即led_cdev->dev=class_create(THIS_MODULE, "leds"),这个将在sys目录下产生文件即产生leds类的文件名,第一个参数指定所属的模块,第二个指定了设备的名字。

而接下来的,第二句IS_ERRleds-class)就是判断leds-class是否正确产生。接下来的都是函数指针。leds_class->suspend = led_suspend就是函数指针啦,上面都有具体函数实现。其中suspend()是在设备休眠时调用,resume()是恢复设备时调用。第一个函数suspend()函数的实现,其实它就是调了brightness_set(led_cdev, 0)函数,所以就说说这个函数。这个函数是数据结构体led_classdev里的成员,是指向一个函数,在哪里指向呢?在leds-gpio.c里的那时候,指向gpio_led_set函数,其实实现就是对level变量赋值。而在leds-s3c24xx.c也差不多。总而来说,实现的供能就是设备挂起时候,就level赋一个值,0还是1就根据你的active_low的选择啦。现在说一下resume(),其实也等同上面一样,最终用led_cdev->brightness赋值给level到这里leds_init函数就OK啦,最后通过subsys_initcall(leds_init)使得eds_init在系统启动时候就会被初始化啦。

 总结一下,leds_init函数在系统启动的时候就会被调用。实现的功能就是在sys/class目录下生成leds类目录,还有就是实现挂起和恢复时候,执行brightness_set(led_cdev, *)函数。

接下来就主要剩下led_classdev_register函数。

前面说了就是产生几个文件。其中第一个就是设备节点。该函数第一句

led_cdev->dev = device_create(leds_class, parent, 0, led_cdev, "%s", led_cdev->name);

函数原型

device_create(struct class *class, struct device *parent,dev_t devt, void *drvdata, const char *fmt, ...)


看看代码实现,一个主要函数就是dev = device_create_vargs(class, parent, devt, drvdata, fmt, vargs),就是实现在对应目录下产生设备节点,那个目录,先看看各个参数意思,第一个参数指定所要创建的设备所从属的类,第二个参数是这个设备的父设备,如果没有就指定为NULL,第三个参数是设备号,第四个参数是设备名称,第五个参数是从设备号。我们看一下实参,第一个实参leds_class,这个在哪里出现的呢?在前面leds_init函数的建立类的返回值,所以其实就是在前面的leds类下建立设备文件节点。

接下来的一句也至关重要,device_create_file函数,添加属性文件,添加了几个文件,我们就那其中一

个dev_attr_brightness来讲,这个属性在哪里实现?上面有 DEVICE_ATTR(brightness, 0644, led_brightness_show, led_brightness_store),其中后面两个参数就是实现对该属性文件的读写操作,两个操作都有具体函数实现。说具体点就是,对该属性文件读操作时候,即使用cat命令对该属性文件操作,内核会自动调用led_brightness_show。同理,使用echo命令就调用led_brightness_store函数现在我们就看看这些文件是会放在哪里。

现在先把代码剖出来:

int led_classdev_register(struct device *parent, struct led_classdev *led_cdev)
{
led_cdev->dev = device_create(leds_class, parent, 0, led_cdev,
     "%s", led_cdev->name);

rc = device_create_file(led_cdev->dev, &dev_attr_brightness);

}

其实这个函数使用EXPORT_SYMBOL_GPL(led_classdev_register)导出来,在leds-gpio.c和leds-s3c24xx.c中调用。

我们跟踪一下led_classdev_register函数。在leds-gpio.c就有调用到,如下

侦测函数:

 gpio_led_probe(struct platform_device *pdev)

调用:

  ret = create_gpio_led(&pdata->leds[i], &leds_data[i],  &pdev->dev, pdata->gpio_blink_set);


create_gpio_led(const struct gpio_led *template,struct gpio_led_data *led_dat, struct device *parent,int (*blink_set)(unsigned, unsigned long *, unsigned long *))

就调用 ret = led_classdev_register(parent, &led_dat->cdev);

前面我们说了建立设备节点,现在再细讲一下。我们主要看看parent指向的是pdev->dev。由于pdev是平台设备,所以得关系到平台设备问题。其中pdev->dev对应平台设备下的设备。所以呢,对于设备节点的设备名,跟踪一下leds-gpio.c代码,就知道:

leds-gpio.c里的cdev就对应上面的led_cdev

led_dat->cdev.name = template->name;  //在函数creat_gpio_led

 template有对应于pdata->leds[i],

struct gpio_led_platform_data *pdata = pdev->dev.platform_data;

简而言之,就是在在gpio_led_probe函数中,获取平台信息platform_data,作为参数传给函数creat_gpio_led的template参数,最后通过该参数付给了led_dat->cdev.name。


所以创建的dev节点的名字由你的平台设备的信息决定的。


现在再来看看在哪里生成属性文件,看函数

device_create_file(led_cdev->dev, &dev_attr_brightness);

主要是看参数led_cdev->dev,这个又是指向哪里,其实就是建立设备节点时候的返回值,可以看看上面。所以就在设备节点目录下建立属性文件,当然后面几个建立属性文件都一样。


说道这里,led-class.c就完啦,剩下没讲的函数要不就是属性读写函数,要不就是卸载函数,对于属性文件就到我们后面的移植篇再做讲解。总结一下,一般类来说,用class_creat加你类,在device_creat在类目录下建立设备文件,还可以在设备节点下建立属性文件,实现对设备的操作,但是该操作一般就是读写,是通过命令实现的。


接下来就差leds-gpio.c(leds-s3c24xx.c和leds_gpio.c)是一样,代码页差不多,,里面主要就是platform模型,即涉及到存放什么硬件资源内核,怎么存放,然后我们又怎么去取出来。。。。。。。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值