首页 > 系统相关 >linux模块驱动编写示例

linux模块驱动编写示例

时间:2023-05-04 18:45:19浏览次数:44  
标签:struct ircut 示例 ret in2 模块 linux gpio printk

2023-05-04

关键字:


该驱动可通过dts配置属性,可在/dev创建节点,支持对/dev下的节点进行标准IO读写以及ioctl读写。

dts如下:

ir_cut {
    status = "okay";
    compatible = "chorm, ir-cut";
    gpios = <&gpf 0 GPIO_ACTIVE_HIGH>, <&gpf 1 GPIO_ACTIVE_LOW>;
};

 

驱动代码如下:

#include <linux/uaccess.h>
#include <linux/cdev.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/platform_device.h>

enum GPIO_FLAGS {
    ACTIVE_HIGH = 0x00,
    ACTIVE_LOW,
};

struct Pin {
    int gpio;
    enum GPIO_FLAGS flag;
};

struct IrCut {
    struct Pin in1;
    struct Pin in2;
    dev_t devid;
    struct cdev* cdev;
    struct class* clz;
    struct device* dev;
};

struct IrCut* gircut;

static int ircut_open(struct inode* node, struct file* fp)
{
    printk("ircut_open()\n");
    int major = MAJOR(node->i_rdev);
    int minor = MINOR(node->i_rdev);
    //fp->private_data = NULL;
    return 0;
}

static int ircut_release(struct inode* node, struct file* fp)
{
    printk("ircut_release()\n");
    return 0;
}

static ssize_t ircut_read(struct file* fp, char __user* buf, size_t cnt, loff_t* offset)
{
    printk("ircut_read()\n");
    int in1 = gpio_get_value(gircut->in1.gpio);
    int in2 = gpio_get_value(gircut->in2.gpio);
    if(gircut->in1.flag == ACTIVE_LOW)
    {
        in1 = !in1;
    }

    if(gircut->in2.flag == ACTIVE_LOW)
    {
        in2 = !in2;
    }

    printk("in1:%s, in2:%s\n", (in1 ? "active" : "inactive"), (in2 ? "active" : "inactive"));

    return 0;
}

static ssize_t ircut_write(struct file* fp, const char __user* buf, size_t len, loff_t* offset)
{
    printk("ircut_write()\n");
    if(len != 4)
    {
        printk("Usage:\n\t1 0\n\tactive in1, inactive in2\n");
        return len;
    }

    char kbuf[5] = {0};
    int ret = copy_from_user(kbuf, buf, len);
    if(ret)
    {
        printk("io error\n");
        return -1;
    }

    int in1, in2;
    if(kbuf[0] == '0' || kbuf[0] == '1')
    {
        in1 = kbuf[0] - '0';
        if(kbuf[1] == ' ')
        {
            if(kbuf[2] == '0' || kbuf[2] == '1')
            {
                in2 = kbuf[2] - '0';
                gpio_set_value(gircut->in1.gpio, (gircut->in1.flag == ACTIVE_LOW ? !in1 : in1));
                gpio_set_value(gircut->in2.gpio, (gircut->in2.flag == ACTIVE_LOW ? !in2 : in2));
            }
        }
    }

    return len;
}

static long ircut_ioctl(struct file* fp, unsigned int cmd, unsigned long arg)
{
    printk("ircut_ioctl()\n");

    unsigned long __user* argp = (unsigned long __user*)arg;
    void* data_struct = (void*)kzalloc(10, 1);
    int ret = copy_from_user(data_struct, argp, 10);

    ret = copy_to_user(arg, data_struct, 10);

    return 0;
}

const struct file_operations ircut_fops = {
    .owner = THIS_MODULE,
    .open = ircut_open,
    .release = ircut_release,
    .read = ircut_read,
    .write = ircut_write,
    .unlocked_ioctl = ircut_ioctl
};

static int ircut_probe(struct platform_device *pdev)
{
    int ret;
    struct device_node* node;
    struct IrCut* ircut;
    gircut = NULL;
    printk("ircut_probe()\n");

    //initialize, parse the dts
    node = pdev->dev.of_node;
    ircut = (struct IrCut*)kzalloc(sizeof(struct IrCut), GFP_KERNEL);
    if(ircut == NULL)
    {
        printk("[fail] No mem\n");
        return ENOMEM;
    }
    memset(ircut, 0, sizeof(struct IrCut));
 
    enum of_gpio_flags flg;
    ret = of_get_named_gpio_flags(node, "gpios", 0, &flg);
    if(ret < 0)
    {
        printk("Can't get gpio[0]\n");
        goto ERR1;
    }
    ircut->in1.gpio = ret;
    ircut->in1.flag = (flg == 0 ? ACTIVE_HIGH : ACTIVE_LOW);
    
    ret = of_get_named_gpio_flags(node, "gpios", 1, &flg);
    if(ret < 0)
    {
        printk("Can't get gpio[1]\n");
        goto ERR1;
    }
    ircut->in2.gpio = ret;
    ircut->in2.flag = (flg == 0 ? ACTIVE_HIGH : ACTIVE_LOW);

    //setup, request the gpios
    ret = gpio_is_valid(ircut->in1.gpio);
    if(!ret)
    {
        printk("gpio[0] invalid\n");
        goto ERR1;
    }

    ret = gpio_request(ircut->in1.gpio, "ircut-in1");
    if(ret < 0)
    {
        printk("request gpio[0] failed:%d\n", ret);
        goto ERR1;
    }

    ret = gpio_direction_output(ircut->in1.gpio, (ircut->in1.flag == ACTIVE_HIGH ? 0 : 1)); //inactive in default
    if(ret < 0)
    {
        printk("gpio[0] set failed:%d\n", ret);
        goto ERR2;
    }

    ret = gpio_is_valid(ircut->in2.gpio);
    if(!ret)
    {
        printk("gpio[1] invalid\n");
        goto ERR2;
    }

    ret = gpio_request(ircut->in2.gpio, "ircut-in2");
    if(ret < 0)
    {
        printk("request gpio[1] failed:%d\n", ret);
        goto ERR2;
    }

    ret = gpio_direction_output(ircut->in2.gpio, (ircut->in2.flag == ACTIVE_HIGH ? 0 : 1)); //inactive too
    if(ret < 0)
    {
        printk("gpio[1] set failed:%d\n", ret);
        goto ERR3;
    }

    printk("in1.gpio:%d, flag:%d\n", ircut->in1.gpio, ircut->in1.flag);
    printk("in2.gpio:%d, flag:%d\n", ircut->in2.gpio, ircut->in2.flag);

    //interface, provide operate handle
    ret = alloc_chrdev_region(&ircut->devid, 0, 1, "ir-cut");
    if(ret)
    {
        printk("alloc devid failed:%d\n", ret);
        goto ERR3;
    }

    ircut->cdev = cdev_alloc();
    if(ircut->cdev == NULL)
    {
        printk("alloc cdev failed\n");
        goto ERR4;
    }

    cdev_init(ircut->cdev, &ircut_fops);

    ret = cdev_add(ircut->cdev, ircut->devid, 1);
    if(ret)
    {
        printk("cdev add failed:%d\n", ret);
        goto ERR4;
    }

    ircut->clz = class_create(THIS_MODULE, "ircut_clz");
    if(ircut->clz == NULL)
    {
        printk("create class failed\n");
        goto ERR4;
    }

    ircut->dev = device_create(ircut->clz, NULL, ircut->devid, NULL, "ircut_dev");
    if(ircut->dev == NULL)
    {
        printk("create device failed\n");
        goto ERR5;
    }


    pdev->dev.platform_data = ircut; //record the private data
    gircut = ircut;
    printk("ircut init success\n");

    return 0;

ERR5:
    class_destroy(ircut->clz);

ERR4:
    unregister_chrdev_region(ircut->devid, 1);

ERR3:
    gpio_free(ircut->in2.gpio);

ERR2:
    gpio_free(ircut->in1.gpio);

ERR1:
    kfree(ircut);

    return EIO;
}

static int ircut_remove(struct platform_device *pdev)
{
    printk("ircut_remove()\n");

    if(pdev->dev.platform_data == NULL)
    {
        return 0;
    }

    struct IrCut* ircut = (struct IrCut*)pdev->dev.platform_data;
    gpio_free(ircut->in1.gpio);
    gpio_free(ircut->in2.gpio);
    printk("gpio freed\n");

    device_destroy(ircut->clz, ircut->devid);
    class_destroy(ircut->clz);
    cdev_del(ircut->cdev);
    unregister_chrdev_region(ircut->devid, 1);
    printk("cdev unregistered\n");

    gircut = NULL;

    return 0;
}

static const struct of_device_id ircut_of_table[] = {
    { .compatible = "chorm, ir-cut" },
    {}
};

static struct platform_driver drv_stu = {
    .probe = ircut_probe,
    .remove = ircut_remove,
    .driver = {
        .name = "ir-cut",
        .of_match_table = ircut_of_table,
    },
};
module_platform_driver(drv_stu);

MODULE_AUTHOR("chorm@lchorm.com");
MODULE_DESCRIPTION("Driver for learing");
MODULE_LICENSE("GPL");

 

更多示例程序参见:

  https://files.cnblogs.com/files/chorm590/linux%E6%A8%A1%E5%9D%97%E9%A9%B1%E5%8A%A8%E7%A4%BA%E4%BE%8B%E4%BB%A3%E7%A0%81_202305041840.zip 

 


 

标签:struct,ircut,示例,ret,in2,模块,linux,gpio,printk
From: https://www.cnblogs.com/chorm590/p/15247701.html

相关文章

  • linux下添加简单的开机自启动脚本
    在rc.local脚本中添加开机自启动程序1、编辑文件/etc/rc.localvim/ect/rc.local2、编辑完/etc/rc.local需要添加执行权限,否则调用失败chmod+x/ect/rc.d/rc.local3、/ect/rc.local和/ect/rc.d/rc.local是软链接关系[root@localhost~]#ll/etc/rc.locallrwxrwxrwx.1......
  • schema模块简介 - 验证数据类型
    目录1schema模块简介2快速上手1.给Schema类传入类型(int、str、float等)2.给Schema类传入可调用的对象(函数、带__call__的类等)3.给Schema类传入带有validate方法的对象4.给Schema类传入容器对象(list、tuple、set等)5.给Schema传入一个字典对象(大部分使用Schema的场景都是传入......
  • 恒创科技:Windows与 Linux 云服务器差异解释
    ​选择云服务器时,重要的是要确定服务器的操作系统。不过,要做出适合您的选择,您需要了解Windows和Linux云服务器之间的主要区别。以下内容旨在提供有关性能、使用情况、安全性、支持和选择这些操作系统的其他方面的相关信息。表现与Windows云服务器相比,Linux可以......
  • Linux系统
    Liunx系统是一个免费,开源,基于Posix和Unix的多用户,多任务,支持多线程和多CPU的操作系统。(免费开源多用户的操作系统)免费:使用这个系统不要钱开源:开放系统源代码,Liunx基本常用命令suroot/123456:超级用户ifconfig:查看IP地址cd /:根目录cd ~:家目录cd../:多级目录ll......
  • Linux搭建JDK环境
    一、上传linux版本的JDK包至服务器本次实验上传的linux目录为/usr/local/develop/JDK。二、解压JDK压缩包tar-zvxfjdk-8u351-linux-x64.tar.gz解压后详情如下三、配置JDK环境变量3.1、打开profile文件,配置jdk的环境变量sudovi/etc/profile3.2、在文件尾......
  • C++获取阿里巴巴1688中国站店铺详情 API 接口返回值示例说明
    ​C++(cplusplus)是一种计算机高级程序设计语言,由C语言扩展升级而产生,最早于1979年由本贾尼·斯特劳斯特卢普在AT&T贝尔工作室研发。C++既可以进行C语言的过程化程序设计,又可以进行以抽象数据类型为特点的基于对象的程序设计,还可以进行以继承和多态为特点的面向对象的程序设计。......
  • 从七个方面聊聊Linux到底强在哪
    从事计算机相关行业的同学不难发现,身边总有一些朋友在学习linux,有的开发同学甚至自己的电脑就是它。经常听他们说linux如何好用等等。那么linux到底好在那里,能让大家如此喜欢。这也是我经常问自己的一个问题。下面我将通过以下七点来为大家阐述linux的巨大优势。 下面我将通过......
  • 深入理解 Linux 的 I/O 系统
    点击上方“C语言与CPP编程”,选择“关注/置顶/星标公众号”干货福利,第一时间送达!传统的SystemCallI/O在Linux系统中,传统的访问方式是通过write()和read()两个系统调用实现的,通过read()函数读取文件到到缓存区中,然后通过write()方法把缓存中的数据输出到网络端口。......
  • Python时间模块time
    时间模块-time#time()获取本地时间戳#ctime()获取本地时间字符串(参数是时间戳,默认当前)#localtime()获取本地时间元组(参数是时间戳,默认当前)#mktime()通过时间元组获取时间戳(参数是时间元组)#asctime()通过时间元组获取时......
  • ARM Linux安装node.js和node-red
    SA515安装node.js和node-red1.先安装node.js直接官网下载ARMv7版可执行文件:https://nodejs.org/en/download当前使用的最新版本:https://nodejs.org/dist/v18.16.0/node-v18.16.0-linux-armv7l.tar.xz2.导入开发板,这里以/usrdata为例adbpushnode-v18.16.0-linux-armv7l......