初探linux子系统集之led子系统(二)【转】
本文转载自:http://blog.csdn.net/eastmoon502136/article/details/37606487
巴西世界杯,德国7比1东道主,那个惨不忍睹啊,早上起来看新闻,第一眼看到7:1还以为点球也能踢成这样,后来想想,点球对多嘛6比1啊,接着就是各种新闻铺天盖地的来了。其实失败并没有什么,人生若是能够成功一次,那么再多的失败也是值得的,脚踏实地,失败了再次爬起来。很多时候,在一切都那么顺利的情况下,反而觉得没有啥意思,有一些挑战,一些失败,一些打击,接着,就会很努力很努力地去做,慢慢地就会被自己征服,很喜欢这种感受,废话说多了,还是接着研究led子系统吧。
对于led子系统中,有那么多得trigger,下面就来简单了解下。
1、default-on
- <pre name="code" class="html">static voiddefon_trig_activate(struct led_classdev *led_cdev)
- {
- led_set_brightness(led_cdev,led_cdev->max_brightness);
- }
- static structled_trigger defon_led_trigger = {
- .name = "default-on",
- .activate = defon_trig_activate,
- };
- static voiddefon_trig_activate(struct led_classdev *led_cdev)
- {
- led_set_brightness(led_cdev,led_cdev->max_brightness);
- }
- static structled_trigger defon_led_trigger = {
- .name = "default-on",
- .activate = defon_trig_activate,
- };
Default-on主要是设置led为最大亮度。
2、backlight
- struct bl_trig_notifier {
- structled_classdev *led; //led子系统设备
- intbrightness; //亮度
- intold_status;
- structnotifier_block notifier; //内核通知链
- unsignedinvert;
- };
- static struct led_trigger bl_led_trigger ={
- .name = "backlight",
- .activate = bl_trig_activate,
- .deactivate = bl_trig_deactivate
- };
- static void bl_trig_activate(structled_classdev *led)
- {
- intret;
- structbl_trig_notifier *n;
- n= kzalloc(sizeof(struct bl_trig_notifier), GFP_KERNEL);
- led->trigger_data= n;
- if(!n) {
- dev_err(led->dev,"unable to allocate backlight trigger\n");
- return;
- }
- ret= device_create_file(led->dev, &dev_attr_inverted);
- if(ret)
- gotoerr_invert;
- n->led= led;
- n->brightness= led->brightness;
- n->old_status= UNBLANK;
- n->notifier.notifier_call= fb_notifier_callback;
- ret= fb_register_client(&n->notifier);
- if(ret)
- dev_err(led->dev,"unable to register backlight trigger\n");
- return;
- err_invert:
- led->trigger_data= NULL;
- kfree(n);
- }
其中fb_register_client注册到了framebuffer中的fb_notifier_list中,一旦framebuffer驱动中有事件,就会调用内核通知链中注册好的函数fb_notifier_callback。
关于内核通知链,这里就插播一曲来自网络的摘抄了:
大多数内核子系统都是相互独立的,因此某个子系统可能对其它子系统产生的事件感兴趣。为了满足这个需求,也即是让某个子系统在发生某个事件时通知其它的子系统,Linux内核提供了通知链的机制。通知链表只能够在内核的子系统之间使用,而不能够在内核与用户空间之间进行事件的通知。
通知链表是一个函数链表,链表上的每一个节点都注册了一个函数。当某个事情发生时,链表上所有节点对应的函数就会被执行。所以对于通知链表来说有一个通知方与一个接收方。在通知这个事件时所运行的函数由被通知方决定,实际上也即是被通知方注册了某个函数,在发生某个事件时这些函数就得到执行。其实和系统调用signal的思想差不多。
通知链技术可以概括为:事件的被通知者将事件发生时应该执行的操作通过函数指针方式保存在链表(通知链)中,然后当事件发生时通知者依次执行链表中每一个元素的回调函数完成通知。
- static int fb_notifier_callback(struct notifier_block *p,
- unsignedlong event, void *data)
- {
- structbl_trig_notifier *n = container_of(p,
- structbl_trig_notifier, notifier);
- struct led_classdev*led = n->led;
- struct fb_event*fb_event = data;
- int *blank =fb_event->data;
- int new_status =*blank ? BLANK : UNBLANK;
- switch (event) {
- case FB_EVENT_BLANK :
- if(new_status == n->old_status)
- break;
- if((n->old_status == UNBLANK) ^ n->invert) {
- n->brightness= led->brightness;
- led_set_brightness(led,LED_OFF);
- } else {
- led_set_brightness(led,n->brightness);
- }
- n->old_status= new_status;
- break;
- }
- return 0;
- }
如果触发了FB_EVENT_BLANK,那么就执行相应的操作。
3、timer
- static structled_trigger timer_led_trigger = {
- .name = "timer",
- .activate =timer_trig_activate,
- .deactivate =timer_trig_deactivate,
- };
- static voidtimer_trig_activate(struct led_classdev *led_cdev)
- {
- int rc;
- led_cdev->trigger_data= NULL;
- rc =device_create_file(led_cdev->dev, &dev_attr_delay_on);
- if (rc)
- return;
- rc =device_create_file(led_cdev->dev, &dev_attr_delay_off);
- if (rc)
- gotoerr_out_delayon;
- led_blink_set(led_cdev,&led_cdev->blink_delay_on,
- &led_cdev->blink_delay_off);
- led_cdev->trigger_data= (void *)1;
- return;
- err_out_delayon:
- device_remove_file(led_cdev->dev,&dev_attr_delay_on);
- }
当某个led_classdev与之连接后,这个触发器会在/sys/class/leds/<device>/下创建两个文件delay_on和delay_off。用户空间往这两个文件中写入数据后,相应的led会按照设置的高低电平的时间(ms)来闪烁。如果led_classdev注册了硬件闪烁的接口led_cdev->blink_set就是用硬件控制闪烁,否则用软件定时器来控制闪烁。
4、heatbeat
- static structled_trigger heartbeat_led_trigger = {
- .name = "heartbeat",
- .activate =heartbeat_trig_activate,
- .deactivate = heartbeat_trig_deactivate,
- };
- structheartbeat_trig_data {
- unsigned int phase;
- unsigned int period;
- struct timer_listtimer;
- };
- static voidheartbeat_trig_activate(struct led_classdev *led_cdev)
- {
- structheartbeat_trig_data *heartbeat_data;
- heartbeat_data =kzalloc(sizeof(*heartbeat_data), GFP_KERNEL);
- if (!heartbeat_data)
- return;
- led_cdev->trigger_data= heartbeat_data;
- setup_timer(&heartbeat_data->timer,
- led_heartbeat_function, (unsigned long)led_cdev);
- heartbeat_data->phase= 0;
- led_heartbeat_function(heartbeat_data->timer.data);
- }
设置了heartbeat_data->phase,然后调用led_heartbeat_function。
- static voidled_heartbeat_function(unsigned long data)
- {
- struct led_classdev*led_cdev = (struct led_classdev *) data;
- structheartbeat_trig_data *heartbeat_data = led_cdev->trigger_data;
- unsigned longbrightness = LED_OFF;
- unsigned long delay =0;
- /* acts like anactual heart beat -- ie thump-thump-pause... */
- switch(heartbeat_data->phase) {
- case 0:
- /*
- * The hyperbolic function below modifies the
- * heartbeat period length in dependency of the
- * current (1min) load. It goes through thepoints
- * f(0)=1260, f(1)=860, f(5)=510,f(inf)->300.
- */
- heartbeat_data->period= 300 +
- (6720<< FSHIFT) / (5 * avenrun[0] + (7 << FSHIFT));
- heartbeat_data->period=
- msecs_to_jiffies(heartbeat_data->period);
- delay =msecs_to_jiffies(70);
- heartbeat_data->phase++;
- brightness =led_cdev->max_brightness;
- break;
- case 1:
- delay =heartbeat_data->period / 4 - msecs_to_jiffies(70);
- heartbeat_data->phase++;
- break;
- case 2:
- delay =msecs_to_jiffies(70);
- heartbeat_data->phase++;
- brightness =led_cdev->max_brightness;
- break;
- default:
- delay =heartbeat_data->period - heartbeat_data->period / 4 -
- msecs_to_jiffies(70);
- heartbeat_data->phase= 0;
- break;
- }
- led_set_brightness(led_cdev,brightness);
- mod_timer(&heartbeat_data->timer,jiffies + delay);
- }
通过定时来实现类似于心跳的led灯。
5、ide-disk
- static voidledtrig_ide_timerfunc(unsigned long data)
- {
- if (ide_lastactivity!= ide_activity) {
- ide_lastactivity =ide_activity;
- /* INT_MAX will seteach LED to its maximum brightness */
- led_trigger_event(ledtrig_ide,INT_MAX);
- mod_timer(&ledtrig_ide_timer,jiffies + msecs_to_jiffies(10));
- } else {
- led_trigger_event(ledtrig_ide,LED_OFF);
- }
- }
- static int __initledtrig_ide_init(void)
- {
- led_trigger_register_simple("ide-disk",&ledtrig_ide);
- return 0;
- }
通过定时器实现类似于硬盘灯的指示。
以上便是led子系统中的trigger的一些简单介绍。
初探linux子系统集之led子系统(二)【转】的更多相关文章
- 初探linux子系统集之led子系统(二)
巴西世界杯,德国7比1东道主,那个惨不忍睹啊,早上起来看新闻,第一眼看到7:1还以为点球也能踢成这样,后来想想,点球对多嘛6比1啊,接着就是各种新闻铺天盖地的来了.其实失败并没有什么,人生若是能够成功 ...
- 初探linux子系统集之led子系统(一)
就像学编程第一个范例helloworld一样,学嵌入式,单片机.fpga之类的第一个范例就是点亮一盏灯.对于庞大的linux系统,当然可以编写一个字符设备驱动来实现我们需要的led灯,也可以直接利用g ...
- 初探linux子系统集之led子系统(一)【转】
本文转载自:http://blog.csdn.net/eastmoon502136/article/details/37569789 就像学编程第一个范例helloworld一样,学嵌入式,单片机.f ...
- 初探linux子系统集之led子系统(三)
世界杯结束了,德国战车夺得了大力神杯,阿根廷最终还是失败了.也许3年,5年,或者10年后,人们就不知道巴西世界杯的亚军是谁,但是总是会记得冠军是谁.就像什么考试,比赛,第一永远会被人们所记住,所以我们 ...
- 初探linux子系统集之led子系统(三)【转】
本文转载自:http://blog.csdn.net/eastmoon502136/article/details/37822837 世界杯结束了,德国战车夺得了大力神杯,阿根廷最终还是失败了.也许3 ...
- 转载-lvs官方文档-Linux服务器集群系统(二)
Linux服务器集群系统(二) LVS集群的体系结构 章文嵩 (wensong@linux-vs.org) 2002 年 4 月 本文主要介绍了LVS集群的体系结构.先给出LVS集群的通用体系结构,并 ...
- 初探linux子系统集之timer子系统(二)
想着博客中还没有翻译过一篇文章,虽然英文水平有限,但是借助google翻译慢慢地翻译出一篇文章也是不错的选择.那就来学习下hrtimer的文档吧,翻译的略搓,可以直接跳过这篇,这里仅作为学习的过程!^ ...
- 初探linux子系统集之i2c子系统(二)
大概也是前年了,一直没有把那个i2c的子系统讲解完,这里偷个懒,把以前整理的i2c相关的知识再梳理一下,做个了结,然后再去学习timer子系统. 先看下i2c在内核中的代码分布: obj-$(CONF ...
- 初探linux子系统集之timer子系统(三)
因为现在的linux虽然还是可以使用低精度的timer,但是趋势是高精度hrtimer,所以上一篇试着翻译一下hrtimer的一些介绍,翻译的不是很好,看来英语还得好好学习啊,下面还是好好学习下lin ...
随机推荐
- ElasticSearch中辅助API常用用法详解
本篇是使用Elasticsearch必不可少的必备知识,并且适用于所有的Rest Api. 返回数据格式化 当在Rest请求后面添加?pretty时,结果会以Json格式化的方式显示.另外,如果添加? ...
- *AtCoder Regular Contest 096F - Sweet Alchemy
$n \leq 50$的树,每个点有权值,现要选点(可多次选一个点)使点数尽量多,如下限制:选的总权值不超过$C \leq 1e9$:$c_i$表示$i$选的次数,$p_i$表示$i$的父亲,那么$c ...
- [C/C++] 结构体内存对齐用法
一.为什么要内存对齐 经过内存对齐之后,CPU的内存访问速度大大提升; 内存空间按照byte划分,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内 ...
- 转载 cc、gcc、g++、CC的区别概括
gcc是C编译器:g++是C++编译器:linux下cc一般是一个符号连接,指向gcc:gcc和g++都是GUN(组织)的编译器.而CC则一般是makefile里面的一个名字,即宏定义,嘿,因为Lin ...
- 搭建Redis环境以及所遇问题(CentOS7+Redis 3.2.8)
一.安装步骤 1. 首先需要安装gcc,把下载好的redis-3.2.8-rc2.tar.gz 放到/usr/local文件夹下 2. 进行解压 tar -zxvf redis-3.2.8-rc2.t ...
- css3 画三角形
/*箭头向上*/ .arrow-up { width:0; height:0; border-left:20px solid transparent; border-right:20px solid ...
- Spring的依赖注入概述
以下内容引用自http://wiki.jikexueyuan.com/project/spring/dependency-injection.html: 每个基于Java应用程序都有几个对象,这些对象 ...
- Linux下使用vi新建文件保存文件时遇到错误:E212: Can't open file for writing
出现E212: Can't open file for writing的问题是由于权限问题导致的,解决方法有以下思路: 1.使用root进行登录,然后再操作. 2.在使用命令时,前面加sudo. 3. ...
- IT部门的KPI该如何制定?
导语:信息化成本.系统开机率.网路不断线时数.系统运行速度.软件开发时间.用户问题处理反应时间.系统品质.用户满意度--哪些指标是可被管理的,能指引IT部门成为一个有价值的.为企业带来效益的部门呢? ...
- 【Todo】RTP/RTCP/RTSP/SIP/SDP 等多媒体传输和会话协议
参考 http://m.blog.csdn.net/article/details?id=6211447