不敲一行代码,实现Linux下LED驱动!

前言


如果要实现一个设备的驱动,一行驱动代码都不用写,这听起来是不是天方夜谭呢?但这并不是不可实现的,因为全世界的内核开发者都非常热心,只要是能写的驱动,他们基本都已经写了。

今天,我们就站在巨人的肩膀上,利用内核开发者已经写好的驱动来实现我们想要的功能,本篇讨论的是LED驱动。

LED驱动


我们以imx6ull pro开发板的板载led为例,其板载了一个可控制的Led2,原理图如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fYKpRJ46-1681393582806)(pic/image-20230412235233576.png)]

LED2进行上拉电阻,另外一个管脚接到了GPIO5_3,因此GPIO5_3输出低电平即可点亮LED。下面说明如何控制该LED。

内核配置:

 Device Drivers  --->
 	[*] LED Support  --->
 		<*>   LED Class Support
 		<*>   LED Support for GPIO connected LEDs
 		[*]   LED Trigger support  --->

我们的LED驱动是基于GPIO的,因此需要打开内核LED驱动的支持。

内核有两个对应的驱动程序,分别是GPIO驱动和LED驱动,基于GPIO的LED驱动调用了GPIO驱动导出的函数。

LED驱动实现代码请参考:drivers/leds/leds-gpio.c,它实现了一个leds类,通过sysfs接口对LED进行控制。

设备树:

leds{
	compatible = "gpio-leds";
	
        led2{
            label = "led2";
            gpios = <&gpio5 3 GPIO_ACTIVE_LOW>;//GPIO_ACTIVE_LOW,代表高电平点亮LED
            default-state = "on";
        };
}

在设备树中创建一个名为leds的节点,compatible"gpio-leds",这样可以匹配到leds-gpio.c的驱动。

然后创建一个子节点,名为led2。需要填三个属性:labelgpiosdefault-state

label:lable是出现在sys目录下的名字,即生成/sys/class/leds/led2

gpios:前两个值指定了该LED所连接的GPIO。第三个值可填GPIO_ACTIVE_HIGHGPIO_ACTIVE_LOWGPIO_ACTIVE_HIGH代表高电平点亮LED,GPIO_ACTIVE_LOW代表低电平点亮LED。

这里注意了,gpios属性的第三个参数,代表该gpio点亮LED是需要高电平还是低电平,注意是点亮LED,细品

default-state:on代表默认情况LED是点亮的,off代表默认LED熄灭

这里又注意了,当defalut-state为on时,实际上gpio输出的电平,就是gpios属性中指定的点亮LED时的电平

设备树配置好后,编译并更换dtb,然后重启开发板。可以看到/sys/class/leds/led2目录:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6k4KDgtF-1681393582807)(pic/image-20230412233937341.png)]

/sys/class/leds/led2/目录下有一个brightnes文件,可以通过echo cat的方式查看和修改LED的亮度。因为LED连接在GPIO上,所以亮度只有0和1,在本文示例的led2中,0表示点亮,1表示熄灭。

点亮LED:

echo 0 > /sys/class/leds/led2/brightness

熄灭LED:

echo 1 > /sys/class/leds/led2/brightness

应用层控制


除了可以在shell中通过echo/cat的方式控制Led,我们也可以在写一个应用层程序来操作/sys/class/leds/下的节点,应用层代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include <string.h>

#include <sys/stat.h>
#include <sys/types.h>

#define LED_DEV_PATH "/sys/class/leds/led%d/brightness"
#define ON
#define OFF

int fs4412_set_led(unsigned int lednum, unsigned int mode)
{
	int fd;
    int ret;
    char devpath[128];
    char *on = "1\n";
    char *off = "0\n";
    char *m = NULL;
    
    snprintf(devpath, sizeof(devpath), LED_DEV_PATH, lednum);
    fd = open(devpath, O_WRONLY);
    if (fd == -1) {
		perror("fsled->open");
        return -1;
	}
    
    if (mode == ON)
        	m = on;
    else
        	m = off;
    
    ret = write(fd, m, strlen(m));
    if (ret == -1) {
		perror("fsled->wrtie");
        close(fd);
        return -1;
    }
    
    close(fd);
    return 0;

}

int main(int argc, char *argv[])
{
	unsigned int lednum = 2;
    
    while(1){
		fs4412_set_led(lednum, on);
        usleep(500000);
        fs4412_set_led(lednum, OFF);
		usleep(500000);
        
        lednum++;
        if (lednum > 5)
            	lednum = 2;
	}

	return 0;
}

上述应用层代码执行后,led2会闪烁。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是一个简单的嵌入式Linux LED驱动程序的示例代码: ```c #include <linux/module.h> #include <linux/init.h> #include <linux/gpio.h> #include <linux/fs.h> #include <linux/cdev.h> #include <linux/device.h> #define DEVICE_NAME "led" #define LED_PIN 23 static dev_t dev_num; static struct cdev *led_cdev; static struct class *led_class; static struct device *led_device; static int led_open(struct inode *inode, struct file *file) { return 0; } static int led_release(struct inode *inode, struct file *file) { return 0; } static ssize_t led_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos) { int value; if (count != 1 || (*buf != '0' && *buf != '1')) return -EINVAL; value = (*buf == '0') ? 0 : 1; gpio_set_value(LED_PIN, value); return 1; } static const struct file_operations led_fops = { .owner = THIS_MODULE, .open = led_open, .release = led_release, .write = led_write, }; static int __init led_init(void) { int ret; ret = gpio_request_one(LED_PIN, GPIOF_OUT_INIT_LOW, "LED"); if (ret) { pr_err("gpio_request_one failed\n"); return ret; } ret = alloc_chrdev_region(&dev_num, 0, 1, DEVICE_NAME); if (ret) { pr_err("alloc_chrdev_region failed\n"); goto err1; } led_cdev = cdev_alloc(); if (!led_cdev) { pr_err("cdev_alloc failed\n"); ret = -ENOMEM; goto err2; } cdev_init(led_cdev, &led_fops); ret = cdev_add(led_cdev, dev_num, 1); if (ret) { pr_err("cdev_add failed\n"); goto err3; } led_class = class_create(THIS_MODULE, DEVICE_NAME); if (IS_ERR(led_class)) { pr_err("class_create failed\n"); ret = PTR_ERR(led_class); goto err4; } led_device = device_create(led_class, NULL, dev_num, NULL, DEVICE_NAME); if (IS_ERR(led_device)) { pr_err("device_create failed\n"); ret = PTR_ERR(led_device); goto err5; } return 0; err5: class_destroy(led_class); err4: cdev_del(led_cdev); err3: kfree(led_cdev); err2: unregister_chrdev_region(dev_num, 1); err1: gpio_free(LED_PIN); return ret; } static void __exit led_exit(void) { device_destroy(led_class, dev_num); class_destroy(led_class); cdev_del(led_cdev); unregister_chrdev_region(dev_num, 1); gpio_free(LED_PIN); } module_init(led_init); module_exit(led_exit); MODULE_AUTHOR("Your Name"); MODULE_DESCRIPTION("LED driver"); MODULE_LICENSE("GPL"); ``` 这个驱动程序使用GPIO 23控制一个LED灯。在打开设备节点文件并写入“0”或“1”时,LED灯将相应地开启或关闭。该驱动程序使用字符设备节点,用户可以像写入文件一样控制LED灯的状态。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

嵌入式Linux充电站

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

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

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

打赏作者

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

抵扣说明:

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

余额充值