当前位置: 首页 > article >正文

(RK3566驱动开发 - 1).pinctrl和gpio子系统

一.设备树

pinctrl部分可以参考 rockchip 官方的绑定文档 :kernel/Documentation/devicetree/bindings/pinctrl

                PIN_BANK:引脚所属的组 - 本次例程使用的是 GPIO3_A1 这个引脚,所以所属的组为 3;

        PIN_BANK_IDX:引脚的索引号,可在 kernel/include/dt-bindings/pinctrl/rockchip.h 中查看到 PA1 的宏定义;

        MUX:引脚的复位功能,同样在绑定文档中的 pinctrl 中可以看到,当 MUX 为 0 的时候表示复用为 gpio ,其余表示其他的复用功能。也可在 rockchip.h 中使用宏 RK_FUNC_GPIO 

        &phandle:电器属性,可在泰山派的 .tspi-rk3566-user-v10-linux.dtb.dts.tmp 中查看

        pcfg_pull_none 为无上下拉

(1).流程图

(2).设备树代码

二.驱动部分

(1).流程图

(2).驱动代码

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_gpio.h>
// #include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
 
 
#define BEEP_CNT     1                    /* 设备号个数 */
#define BEEP_NAME    "beep"              /* 名字 */
#define BEEPOFF          0               /* 关 */
#define BEEPON           1               /* 开 */
 
 
/* beep设备结构体 */
struct beep_dev
{
    dev_t devid;                /* 设备号 */
    struct cdev cdev;           /* cdev */
    struct class *class;        /* 类 */
    struct device *device;      /* 设备 */
    int major;                  /* 主设备号 */
    int minor;                  /* 次设备号 */
    struct device_node *nd;     /* 设备结点 */
    int beep_gpio;              /* beep所使用的 gpio 编号 */
};
 
 
/* led设备 */
struct beep_dev beepdev;
 
/**
 * @description:            打开设备
 * @param - inode   :       传递给驱动的inode
 * @param - filp    :       设备文件
 * @return          :       0 成功,其他 失败      
*/
static int beep_open(struct inode *inode,struct file *filp)
{
    /* 设置私有属性 */
    filp->private_data = &beepdev; 
 
    return 0;
}
 
 
/**
 * @description:            从设备读取数据
 * @param - filp    :       要打开的设备文件(文件描述符)
 * @param - buf     :       返回给用户空间的数据缓冲区
 * @param - cnt     :       要读取的数据长度
 * @param - offt    :       相对于文件首地址的偏移
 * @return          :       读取的字节数,如果为负值,则为失败
*/
static ssize_t beep_read(struct file *filp,char __user *buf,size_t cnt,loff_t *offt)
{
    return 0;
}
 
 
/**
 * @description:            向设备写数据
 * @param - filp    :       设备文件,表示打开的文件描述符
 * @param - buf     :       要写给设备的数据
 * @param - cnt     :       要写入的数据长度
 * @param - offt    :       写入的字节数,如果为负值,则为失败
*/
static ssize_t beep_write(struct file *filp,const char __user *buf,size_t cnt,loff_t *offt)
{
    int retvalue;
    unsigned char databuf[1];
    unsigned char beepstat;
    struct beep_dev *dev = filp->private_data;
 
    retvalue = copy_from_user(databuf,buf,cnt);
    if(0 > retvalue)
    {
        printk("kernel write failed!\r\n");
        return -EFAULT;
    }
 
    beepstat = databuf[0];
    printk("beepstat : %d\r\n",beepstat);

    if(beepstat == BEEPON)
    {
        gpio_set_value(dev->beep_gpio,1);    //打开LED灯
    }
    else if(beepstat == BEEPOFF)
    {
        gpio_set_value(dev->beep_gpio,0);    //关闭LED
    }
 
    return 0;
}
 
 
 
/**
 * @description:        关闭/释放设备
 * @param - filp    :   要关闭的设备文件(文件描述符)
 * @return          :   0 成功,其他 失败  
*/
static int beep_release(struct inode *inode,struct file *filp)
{
    return 0 ;
}
 
 
/* 绑定设备操作函数 */
static struct file_operations beep_fops = 
{
    .owner = THIS_MODULE,
    .open = beep_open,
    .read = beep_read,
    .write = beep_write,
    .release = beep_release,
};
 
 
/**
 * @description:        驱动入口函数
 * @param           :   无
 * @return          :   无
*/
static int __init beep_init(void)
{
    int ret = 0;
    printk("enter beep_init\r\n");
 
    /* 设置LED所使用的beep */
    /* 1.从设备数中获取设备节点:beep */
    beepdev.nd = of_find_node_by_path("/beep");
    if(NULL == beepdev.nd)
    {
        printk("beep node can not found!\r\n");
    }
    else
    {
        printk("beep node has been found!\r\n");
    }
 
    /* 2.获取设备数中的beep属性,得到LED所使用的LED编号 */
    beepdev.beep_gpio = of_get_named_gpio(beepdev.nd,"beep-gpio",0);
    if(0 > beepdev.beep_gpio)
    {
        printk("can not get gpio-beep");
        return -EINVAL;
    }
    printk("gpio-beep num = %d\r\n",beepdev.beep_gpio);
 
    /* 3.初始化beep,默认关闭LED */
    ret = gpio_direction_output(beepdev.beep_gpio,0);
    if(0 > ret)
    {
        printk("can not init beep!\r\n");
    }
 
 
 
    /* 注册字符设备驱动 */
    /* 1.创建设备号 */
    if(beepdev.major)       //若定义了设备号
    {
        beepdev.devid = MKDEV(beepdev.major,0);
        register_chrdev_region(beepdev.devid,BEEP_CNT,BEEP_NAME);
    }
    else                    //没有定义设备号
    {
        alloc_chrdev_region(&beepdev.devid,0,BEEP_CNT,BEEP_NAME);   //申请设备号
        beepdev.major = MAJOR(beepdev.devid);                       //获取主设备号
        beepdev.minor = MINOR(beepdev.devid);                       //获取次设备号
    }
    printk("beep major = %d,minor = %d",beepdev.major,beepdev.minor);
 
    /* 2.初始化cdev */
    beepdev.cdev.owner = THIS_MODULE;
    cdev_init(&beepdev.cdev,&beep_fops);
 
    /* 3.添加一个cdev */
    cdev_add(&beepdev.cdev,beepdev.devid,BEEP_CNT);
 
    /* 4.创建类 */
    beepdev.class = class_create(THIS_MODULE,BEEP_NAME);
    if(IS_ERR(beepdev.class))
    {
        return PTR_ERR(beepdev.class);
    }
 
    /* 5.创建设备 */
    beepdev.device = device_create(beepdev.class,NULL,beepdev.devid,NULL,BEEP_NAME);
    if(IS_ERR(beepdev.device))
    {
        return PTR_ERR(beepdev.device);
    }
 
    return 0;
}
 
 
/**
 * @description:        驱动出口函数
 * @param       :       无
 * @return      :       无
*/
static void __exit beep_exit(void)
{
    printk("enter beep_exit");
    /* 注销字符设备驱动 */
    cdev_del(&beepdev.cdev);        //删除cdev
    unregister_chrdev_region(beepdev.devid,BEEP_CNT);        //注销
 
    device_destroy(beepdev.class,beepdev.devid);
    class_destroy(beepdev.class);
}
 
 
 
module_init(beep_init);
module_exit(beep_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("kaneki");

(3).Makefile

PWD ?= $(shell pwd)

KERNELDIR := /home/linux/RK3566/rk3566_sdk/kernel
CROSS_COMPILE := /home/linux/RK3566/rk3566_sdk/prebuilts/gcc/linux-x86/aarch64/gcc-linaro-6.3.1-2017.05-x86_64_aarch64-linux-gnu/bin/aarch64-linux-gnu-

obj-m += beep.o

CC := $(CROSS_COMPILE)gcc

module:
	make -C $(KERNELDIR) M=$(PWD) ARCH=arm64 modules
	@# -C $(KERNELDIR) 从当前目录切换到内核源码下 借助内核源码 makefile 进行 makefile
	@# M=$(PWD) 只编译当前目录下的驱动文件
	@# ARCH=arm64 指定编译架构
	# $(CC) beep.c -o beep
clean:
	make -C $(KERNELDIR) M=$(PWD) ARCH=arm64 clean

三.应用程序

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

#define BEEP_ON     1
#define BEEP_OFF    0

int main(int argc, char  *argv[])
{
    char *filename;
    char databuf[1];
    int fd,ret;

    if(3 != argc)
    {
        printf("usage : ./%s <dev_path> <1 / 0>\n",argv[0]);
        return -1;
    }

    filename = argv[1];
    databuf[0] = atoi(argv[2]);

    fd = open(filename,O_WRONLY);
    if(0 > fd)
    {
        perror("open failed");
        return -1;
    }


    write(fd,databuf,1);


    return 0;
}


http://www.kler.cn/a/389293.html

相关文章:

  • influxDB 时序数据库安装 flux语法 restful接口 nodjsAPI
  • 更改Ubuntu22.04锁屏壁纸
  • HarmonyOS Next星河版笔记--界面开发(4)
  • GISBox VS ArcGIS:分别适用于大型和小型项目的两款GIS软件
  • 【C++】类与对象的基础概念
  • 深入理解接口测试:实用指南与最佳实践5.0(二)
  • vscode中使用c/c++插件运行c代码后占用c盘空间问题的解决
  • 呼叫中心系统监控预警功能的应用
  • 纯css制作声波扩散动画、js+css3波纹催眠动画特效、【css3动画】圆波扩散效果、雷达光波效果完整代码
  • 【LeetCode】【算法】209. 课程表
  • 蓝桥杯备赛(持续更新)
  • 【C语言刷力扣】66.加一
  • 京东商品SKU信息的“窃听风云”:Python爬虫的幽默之旅
  • 搜维尔科技:【应用】Xsens在荷兰车辆管理局人体工程学评估中的应用
  • 『 Linux 』网络传输层 - TCP(三)
  • 基于百度飞桨paddle的paddlepaddle2.4.2等系列项目的运行
  • Tidb数据恢复
  • [CKS] Create/Read/Mount a Secret in K8S
  • 软考中级 软件设计师 上午考试内容笔记(个人向)Part.3
  • Linux 消息队列
  • go template 模板字符串
  • std::thread线程通知、等待、让渡
  • 绿色能源发展关键:优化风电运维体系
  • 初学Java基础---Day21---正则表达式,日期类,Math类,Random类,System类,Runtime类,大数值运算类,
  • 【cursor添加azure】在cursor中添加azure的openai api
  • 面向对象试题带答案