Linux 内核集成了LED驱动,采用platform框架,我们只需要在编译内核时选择LED驱动,在设备树文件中添加LED节点即可。
按Y使能。保存退出,然后编译内核。
在上面使能LED驱动后,drivers/leds/leds-gpio.c就会被编译。 在leds-gpio.c中有设备匹配表如下:
static const struct of_device_id of_gpio_leds_match[] = { { .compatible = "gpio-leds", }, { }, }; ...... static struct platform_driver gpio_led_driver = { .probe = gpio_led_probe, .remove = gpio_led_remove, .driver = { .name = "leds-gpio", .of_match_table = of_gpio_leds_match, }, }; module_platform_driver(gpio_led_driver); 因此设备树中的compatible属性为gpio-leds的节点都会被leds-gpio.c驱动匹配。在gpio_led_driver中name为leds-gpio,因此在注册驱动之后,/sys/bus/platform/drivers/下会有leds-gpio驱动。其中module_platform_driver(gpio_led_driver);用于注册驱动。其定义在include/linux/platform_device.h中,如下: #define module_platform_driver(__platform_driver) \ module_driver(__platform_driver, platform_driver_register, \ platform_driver_unregister)可以看出是一个宏,依赖于module_driver(),module_driver()定义在include/linux/device.h中,如下:
#define module_driver(__driver, __register, __unregister, ...) \ static int __init __driver##_init(void) \ { \ return __register(&(__driver) , ##__VA_ARGS__); \ } \ module_init(__driver##_init); \ static void __exit __driver##_exit(void) \ { \ __unregister(&(__driver) , ##__VA_ARGS__); \ } \ module_exit(__driver##_exit);由此可知,将module_platform_driver(gpio_led_driver);展开后为:
static int __init gpio_led_driver_init(void) { return platform_driver_register (&(gpio_led_driver)); } module_init(gpio_led_driver_init); static void __exit gpio_led_driver_exit(void) { platform_driver_unregister (&(gpio_led_driver) ); } module_exit(gpio_led_driver_exit);即为platform驱动的注册和删除。
添加dtsled节点如下:
dtsleds { compatible = "gpio-leds"; pinctrl-names = "default"; pinctrl-0 = <&pinctrl_light>; led_red { label = "red"; gpios = <&gpio1 4 GPIO_ACTIVE_LOW>; default-state = "on"; linux,default-trigger = "timer"; }; };编译设备树,重启。