全志 A64 linux 通过设备树写LED驱动(附参考代码)

简介: 开发平台 芯灵思Sinlinx A64内存: 1GB 存储: 4GB详细参数 https://m.tb.cn/h.3wMaSKm开发板交流群 641395230全志A64设备树结构体 #include <linux/of.

开发平台 芯灵思Sinlinx A64

内存: 1GB 存储: 4GB
详细参数 https://m.tb.cn/h.3wMaSKm
开发板交流群 641395230

全志A64设备树结构体

#include <linux/of.h> //设备树里的每个设备及每个设备子节点都用此结构体描述

struct device_node
{
    const char *name;
    const char *type;
    phandle phandle;
    const char *full_name;
    struct property *properties; //属性
    struct property *deadprops; /* removed properties */
    struct device_node *parent; //在设备子节点对象,指向属于的设备对象
    struct device_node *child; //在设备对象,指向子节点
    struct device_node *sibling; //指向同级的下一个对象.
    struct device_node *next; /* next device of same type */ //应是指向device_type是同样的对象
    struct device_node *allnext; /* next in list of all nodes */ ...
};

//下面函数用于获取设备树里的设备节点及设备子节点
extern struct device_node *of_find_node_by_name(struct device_node *from, const char *name);
//通过名字查找相应的设备节点
static inline int of_get_child_count(const struct device_node *np);
//获取指定设备的子节点个数
extern struct device_node *of_find_node_by_path(const char *path);
//通过路径来获取设备节点,可用于获取设备子节点
extern struct device_node *of_find_node_by_type(struct device_node *from, const char *type); //通过指定的device_type来获取设备节点

//下面函数用于获取设备节点或设备子节点的属性

static inline int of_property_read_u32(const struct device_node *np, const char *propname, u32 *out_value)
extern int of_property_read_u32_index(const struct device_node *np, const char *propname, u32 index, u32 *out_value);
extern int of_property_read_u8_array(const struct device_node *np, const char *propname, u8 *out_values, size_t sz);
extern int of_property_read_u16_array(const struct device_node *np, const char *propname, u16 *out_values, size_t sz);
extern int of_property_read_u32_array(const struct device_node *np, const char *propname, u32 *out_values, size_t sz);
extern int of_property_read_u64(const struct device_node *np, const char *propname, u64 *out_value);
extern int of_property_read_string(struct device_node *np, const char *propname, const char **out_string)

首先增加节点,修改dtsi文件。
vim /lichee/linux-3.10/arch/arm64/boot/dts/sun50iw1p1-pinctrl.dtsi

驱动代码:

    #include <linux/module.h>   
    #include <linux/init.h>  
    #include <linux/fs.h>   
    #include <linux/device.h>   
    #include <linux/slab.h>  
    #include <linux/cdev.h>  
    #include <asm/uaccess.h>  
    #include <linux/io.h>  
    #include <linux/of.h>  
    #include <linux/of_gpio.h>  
    #include <linux/gpio.h>  
    #include <linux/sys_config.h>  
      
    #define MY_DEVICE_NAME "my_led_device"  
    // 获取到设备树中到节点  
    static int gpio = -1;   
    int get_irqno_from_node(void)  
    {  
          
        struct gpio_config config;   
        struct device_node *np = of_find_node_by_path("/leds");  
        if(np){  
            printk("find node ok\n");  
        }  
        else{  
            printk("find node failed\n");  
        }  
      
        gpio = of_get_named_gpio_flags(nd, "gpios", i, (enum of_gpio_flags *)&config);// 从设备树中读取gpios的GPIO配置编号和标志  
        if(!gpio_is_valid(gpio)){  
            //判断该 GPIO 编号是否有效,有效gpio_request 则申请占用该 GPIO。如果初始化过程出错,需要调用 gpio_free 来释放之前申请过且成功的 GPIO   
            printk("gpio isn't valid\n");  
            return -1;  
        }  
        if(gpio_request(gpio, "leds") < 0)   
            printk("gpio request failed %d\n", gpio);   
        gpio_direction_output(gpio, 1); //关灯  
              
        return 0;   
      
    }  
      
    static int my_open (struct inode *node, struct file *filp)  
    {  
        if(gpio)   
        {  
            printk("open ok\n");   
        }  
        else   
        {  
            return -EINVAL;  
        }  
        return 0;  
    }  
      
    static ssize_t my_write (struct file *filp, const char __user *buf, size_t size, loff_t *off)  
    {  
        unsigned char val;          
        copy_from_user(&val, buf, 1);  
        printk(" gpl_dat address   0x%x\n",gpl_dat);  
        if (val)  
        {      
            gpio_direction_output(gpio, 0); //关灯  
            printk("led on\n");  
        }  
        else  
        {  
           gpio_direction_output(gpio, 1); //关灯  
            printk("led off\n");  
        }  
      
        return 1;   
    }  
      
      
    static const struct file_operations my_led_fops = {  
        //step 1 :定义file_operations结构体  
        .open = my_open,  
        .write = my_write,      
    };  
      
    //step 1 :  
    static struct class *led_class;  
    static struct cdev *pcdev;      //定义一个cdev指针  
    static dev_t n_dev;            //第一个设备号(包含了主和次)  
    static int __init led_device_init(void)  
    {//step 2 :注册   
        int ret = -1;  
        pcdev = cdev_alloc();//分配cdev结构空间  
        if(pcdev == NULL) {  
            printk(KERN_EMERG" cdev_alloc  error\n");  
            ret = -ENOMEM;   /* 分配失败 */  
            return ret;  
        }  
        //2. 动态申请设备号  
        ret = alloc_chrdev_region(&n_dev, 0 , 2, MY_DEVICE_NAME);  
        if(ret < 0 ) {  
            //释放前面成功的资源  
            kfree(pcdev);                              /*释放cdev结构空间 */  
            printk(KERN_EMERG"alloc_chrdev_region  error\n");  
            return ret;  
        }      
        cdev_init(pcdev, &my_led_fops);     //初始化cdev结构           /* 建立cdev和file_operations之间的连接 */   
        /* 
            或这样初始化cdev结构 
            pcdev->owner = THIS_MODULE; 
            pcdev->ops = &my_led_fops; 
        */  
        ret = cdev_add(pcdev, n_dev, 2) ;// 向内核里面添加一个驱动,注册驱动  
        if(ret < 0 ) {  
            //释放前面成功的资源  
            unregister_chrdev_region(n_dev,  2);       /*  释放前面申请的调和号*/  
            kfree(pcdev);                               /* 释放cdev结构空间 */  
            printk(KERN_EMERG"alloc_chrdev_region  error\n");  
            return ret;  
        }  
      
        /*自动创建设备节点/dev/SinlinxA64_LED*/  
        led_class = class_create(THIS_MODULE, "myled");      
        device_create(led_class, NULL, n_dev, NULL, "SinlinxA64_LED");   
      
        get_irqno_from_node();  
        printk(KERN_EMERG"cdev ok\n");      
        return 0;  
    }  
      
    static void __exit led_device_exit(void)  
    {    //step 2 :注销  
      
        //注销cdev结构  
        cdev_del(pcdev);  
        //释放设备号  
        unregister_chrdev_region(n_dev, 2); /*起始设备号(主、次) 连续的次设备号数量*/  
        //释放cdev结构空间  
        kfree(pcdev);  
          
        device_destroy(led_class, n_dev);  
        class_destroy(led_class);  
        gpio_free(gpio);   
        printk(KERN_EMERG"cdev_del ok\n");  
    }  
      
    module_init(led_device_init);  
    module_exit(led_device_exit);  
    MODULE_LICENSE("GPL");  

参考文章:https://blog.csdn.net/jklinux/article/details/82382066

相关文章
|
1月前
|
Ubuntu Linux Shell
Linux 系统中的代码类型或脚本类型内容
在 Linux 系统中,代码类型多样,包括 Shell 脚本、配置文件、网络配置、命令行工具和 Cron 定时任务。这些代码类型广泛应用于系统管理、自动化操作、网络配置和定期任务,掌握它们能显著提高系统管理和开发的效率。
|
2月前
|
Linux C语言 C++
vsCode远程执行c和c++代码并操控linux服务器完整教程
这篇文章提供了一个完整的教程,介绍如何在Visual Studio Code中配置和使用插件来远程执行C和C++代码,并操控Linux服务器,包括安装VSCode、安装插件、配置插件、配置编译工具、升级glibc和编写代码进行调试的步骤。
362 0
vsCode远程执行c和c++代码并操控linux服务器完整教程
|
3月前
|
Java Linux Python
Linux环境下 代码java调用python出错
Linux环境下 代码java调用python出错
68 4
|
3月前
|
Linux Shell Python
9-7|salt代码在linux机子那个目录
9-7|salt代码在linux机子那个目录
|
3月前
|
Linux 开发者 Python
从Windows到Linux,Python系统调用如何让代码飞翔🚀
【9月更文挑战第10天】在编程领域,跨越不同操作系统的障碍是常见挑战。Python凭借其“编写一次,到处运行”的理念,显著简化了这一过程。通过os、subprocess、shutil等标准库模块,Python提供了统一的接口,自动处理底层差异,使代码在Windows和Linux上无缝运行。例如,`open`函数在不同系统中以相同方式操作文件,而`subprocess`模块则能一致地执行系统命令。此外,第三方库如psutil进一步增强了跨平台能力,使开发者能够轻松编写高效且易维护的代码。借助Python的强大系统调用功能,跨平台编程变得简单高效。
62 0
|
1月前
|
Linux 网络安全 数据安全/隐私保护
Linux 超级强大的十六进制 dump 工具:XXD 命令,我教你应该如何使用!
在 Linux 系统中,xxd 命令是一个强大的十六进制 dump 工具,可以将文件或数据以十六进制和 ASCII 字符形式显示,帮助用户深入了解和分析数据。本文详细介绍了 xxd 命令的基本用法、高级功能及实际应用案例,包括查看文件内容、指定输出格式、写入文件、数据比较、数据提取、数据转换和数据加密解密等。通过掌握这些技巧,用户可以更高效地处理各种数据问题。
95 8
|
1月前
|
监控 Linux
如何检查 Linux 内存使用量是否耗尽?这 5 个命令堪称绝了!
本文介绍了在Linux系统中检查内存使用情况的5个常用命令:`free`、`top`、`vmstat`、`pidstat` 和 `/proc/meminfo` 文件,帮助用户准确监控内存状态,确保系统稳定运行。
266 6
|
1月前
|
Linux
在 Linux 系统中,“cd”命令用于切换当前工作目录
在 Linux 系统中,“cd”命令用于切换当前工作目录。本文详细介绍了“cd”命令的基本用法和常见技巧,包括使用“.”、“..”、“~”、绝对路径和相对路径,以及快速切换到上一次工作目录等。此外,还探讨了高级技巧,如使用通配符、结合其他命令、在脚本中使用,以及实际应用案例,帮助读者提高工作效率。
80 3
|
1月前
|
监控 安全 Linux
在 Linux 系统中,网络管理是重要任务。本文介绍了常用的网络命令及其适用场景
在 Linux 系统中,网络管理是重要任务。本文介绍了常用的网络命令及其适用场景,包括 ping(测试连通性)、traceroute(跟踪路由路径)、netstat(显示网络连接信息)、nmap(网络扫描)、ifconfig 和 ip(网络接口配置)。掌握这些命令有助于高效诊断和解决网络问题,保障网络稳定运行。
73 2
|
16天前
|
Linux Shell
Linux 10 个“who”命令示例
Linux 10 个“who”命令示例
44 14
Linux 10 个“who”命令示例