Linux应用层点亮硬件的LED灯

一 应用层操作硬件的两种方法

应用层想要对底层硬件进行操控,通常可以通过两种方式:

  1. /dev/目录下的设备文件(设备节点);
  2. /sys/目录下设备的属性文件。

具体使用哪种方式需要根据不同功能类型设备进行选择,通常情况下,一般简单地设备会使用sysfs方式操控,其设备驱动在实现时会将设备的一些属性导出到用户空间sysfs文件系统,以属性文件的形式为用户空间提供对这些数据、属性的访问支持,譬如LED、GPIO等。但对于一些较复杂的设备通常会使用设备节点的方式,譬如LCD等、触摸屏、摄像头等。

 1.1 什么是sysfs文件系统

sysfs是一个基于内存的文件系统,同devfs、proc文件系统一样,称为虚拟文件系统;它的作用是将内核信息以文件的方式提供给应用层使用,sysfs文件系统的主要功能便是对系统设备进行管理,它可以产生一个包含所有系统硬件层次的视图。、

 1.2 sysfs与/sys的关系

sysfs文件系统挂载在/sys目录下,启动ALPHA/Mini I.MX6U开发板,进入Linux系统(开发板出厂系统)之后,我们进入到/sys目录下查看,如下所示:

/sys下的子目录 说明 
/sys/devices 这是系统中所有设备存放的目录,也就是系统中的所有设
备在sysfs中的呈现、表达,也是sysfs管理设备的最重要
的目录结构。 
/sys/block 

块设备的存放目录,这是一个过时的接口,按照sysfs的设
计理念,系统所有的设备都存放在/sys/devices目录下,所

以/sys/block目录下的文件通常是链接到/sys/devices目录下的文件。 

/sys/bus 这是系统中的所有设备按照总线类型分类放置的目录结
构,/sys/devices目录下每一种设备都是挂在某种总线下
的,譬如i2c设备挂在I2C总线下。同样,/sys/bus目录下
的文件通常也是链接到了/sys/devices目录。 
/sys/class这是系统中的所有设备按照其功能分类放置的目录结构,
同样该目录下的文件也是链接到了/sys/devices目录。按照
设备的功能划分组织在/sys/class目录下,譬如/sys/class/leds
目录中存放了所有的LED设备,/sys/class/input目录中存放
/sys/dev 这是按照设备号的方式放置的目录结构,同样该目录下的
文件也是链接到了/sys/devices目录。该目录下有很多以主
设备号:次设备号(major:minor)命名的文件,这些文件都
是链接文件,链接到/sys/devices目录下对应的设备。 
/sys/firmware 描述了内核中的固件
/sys/fs 用于描述系统中所有文件系统,包括文件系统本身和按文
件系统分类存放的已挂载点。
/sys/kernel 这里是内核中所有可调参数的位置
/sys/module 这里有系统中所有模块的信息。 
/sys/power 这里是系统中电源选项,有一些属性可以用于控制整个系
统的电源状态。 

系统中所有的设备(对象)都会在/sys/devices体现出来,是sysfs文件系统中最重要的目录结构;而/sys/bus、/sys/class、/sys/dev分别将设备按照挂载的总线类型、功能分类以及设备号的形式将设备组织存放在这些目录中,这些目录下的文件都是链接到了/sys/devices中。

 1.3 LED的控制方式

正点原子 I.MX6U开发板底板上有一颗可被用户控制的LED灯,如下所示:

此LED设备使用的是Linux内核标准LED驱动框架注册而成,在/dev目录下并没有其对应的设备节点,其实现使用sysfs方式控制。进入到/sys/class/leds目录下,如下所示:  

 

这里我们主要关注便是brightness、max_brightness以及trigger三个文件,这三个文件都是LED设备的属性文件:  

  1. brightness:控制LED的亮灭
  2. max_brightness:该属性文件只能被读取,不能写,用于获取LED设备的最大亮度等级。
  3. trigger:触发模式,该属性文件可读可写,读表示获取LED当前的触发模式,写表示设置LED的触发模式。不同的触发模式其触发条件不同,LED设备会根据不同的触发条件自动控制其亮、灭状态,通过cat命令查看该属性文件,可获取LED支持的所有触发模式以及LED当前被设置的触发模式:
  4. 方括号([heartbeat])括起来的表示当前LED对应的触发模式,none表示无触发,常用的触发模式包括none(无触发)、mmc0(当对mmc0设备发起读写操作的时候LED会闪烁)、timer(LED会有规律的一亮一灭,被定时器控制住)、heartbeat(心跳呼吸模式,LED模仿人的心跳呼吸那样亮灭变化)。

 大致控制方式:将设备属性文件进行写入操作

往trigger写入参数可改变LED灯的触发方式  

往brightness写入数据可以开关LED灯,写1为开,写0为关

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#define  LED_TRIGGER "/sys/class/leds/sys-led/trigger"              //LED触发方式属性文件
#define  LED_ON_OFF "/sys/class/leds/sys-led/brightness"       //LED灯的开和关
#define  HELP()  fprintf(stderr, "usage:\n   %s <on|off>\n %s <trigger> <type>\n", argv[0], argv[0])
#define  NONE 0                     //none表示无触发,常用的触发模式包括none(无触发)
#define MMC0 1                      //mmc0(当对 mmc0设备发起读写操作的时候 LED会闪烁)
#define TIMER 2                     //timer(LED会有规律的一亮一灭,被定时器控制住)
#define HEARTBEAT 3          //heartbeat(心跳呼吸模式,LED模仿人的心跳呼吸那样亮灭变化)。


/* 识别从键盘输入的第三个参数*/
int msg_handle(char *buf)
{
    if(!strcmp(buf,"none")){return NONE;}
    if(!strcmp(buf,"mmc0")){return MMC0;}
    if(!strcmp(buf,"timer")){return TIMER;}
    if(!strcmp(buf,"heartbeat")){return HEARTBEAT;} 
    return -1;
}


int main(int argc ,char *argv[])
{
    int fd1 = 0;        //打开控制触发方式的设备属性文件的索引
    int fd2 = 0;        //打开LED亮灭的设备属性文件的索引
    int ret  = 0;
    int write_ret = 0;   //write函数的返回值
    if(argc < 2)    //参数少于2时错误
    {
        HELP();
        exit(-1);
    }
    fd1 = open(LED_TRIGGER,O_RDWR);   //fd1设备文件操作为控制触发方式
    if(fd1<0)
    {
        perror("触发模式:");
    }
    fd2 =  open(LED_ON_OFF,O_RDWR);  //fd2设备文件用于控制开关
    if(fd2<0)
    {
        perror("开关状态:");
    }
    if(!strcmp(argv[1],"on"))   //打开LED灯
    {
        write(fd1,"none",strlen("none"));
        write(fd2,"1",1);
    }
    else if(!strcmp(argv[1],"off"))  //关闭LED灯
    {
        write(fd1,"none",strlen("none"));
        write(fd2,"0",1);
    }

    else if(!strcmp(argv[1],"trigger"))   //改变LED灯的触发方式
    {
        if(argc != 3)
        {
            HELP();
            exit(-1);
        }
        //当参数2为trigger时,运行程序时需要第三参数,none|mmc0|timer|heartbeat
        ret = msg_handle(argv[2]);

        switch (ret)
        {
        case NONE:
            
            break;
        case MMC0:
        
            break;
        case TIMER:
            write_ret =  write(fd1,argv[2],strlen(argv[2]));  //定时器呼吸灯,按固定频率亮灭
            if(write_ret < 0){perror("");}
            break;
        case HEARTBEAT:
            write_ret =  write(fd1,argv[2],strlen(argv[2]));  //呼吸灯
            if(write_ret < 0){perror("");}
            break;

        default:
            HELP();
            break;
        }
    }
    close(fd1);
    close(fd2);
}

程序中定义了两个宏,LED_TRIGGER和LED_BRIGHTNESS,分别对应/sys/class/leds/sys-led/trigger和/sys/class/leds/sys-led/brightness属性文件,HELP()为打印帮助信息

 二 交叉编译

交叉编译器的安装请看主页其他文章

在使用之前先对交叉编译工具的环境进行设置,使用source执行安装目录下的environment-setup-cortexa7hf-neon-poky-linux-gnueabi脚本文件即可,如下所示:

source /opt/fsl-imx-x11/4.1.15-2.1.0/environment-setup-cortexa7hf-neon-poky-linux-gnueabi

 arm-linux-gnueabihf-gcc leds.c -o led就是将leds.c文件交叉编译为开发板的可执行文件

 scp led root@192.168.121.168:/home/root为将led可执行文件拷贝到开发板中

接下来执行led程序测试:

  1. ./led on # 点亮LED
  2. ./ledoff # 熄灭LED
  3. ./led trigger heartbeat # 将LED触发模式设置为heartbeat
  4. ./led trigger timer # 将LED触发模式设置为timer
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

@ChenPi

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值