编写LED灯的驱动,使用GPIO子系统,里面添加按键的中断处理

编写LED灯的驱动,使用GPIO子系统,里面添加按键的中断处理
1.应用程序发送指令控制LED亮灭
2.按键1按下,led1电位反转按键2按下,led2电位反转按键3按下,led3电位反转

内核代码

#include 
#include 
#include 
#include 
#include 
#include 
#include 
struct device_node *dev_irq1;
struct device_node *dev_irq2;
struct device_node *dev_irq3;
struct device_node *dev_led1;
struct device_node *dev_led2;
struct device_node *dev_led3;
struct gpio_desc *gpio1no;
struct gpio_desc *gpio2no;
struct gpio_desc *gpio3no;
struct timer_list mytimer;
unsigned int irqno1,irqno2,irqno3;

void mytimer_func(struct timer_list *timer)
{
    
}
irqreturn_t myirq1_handler(int irq,void *dev)
{
    gpiod_set_value(gpio1no,!gpiod_get_value(gpio1no));
    return IRQ_HANDLED;
}
irqreturn_t myirq2_handler(int irq,void *dev)
{
    gpiod_set_value(gpio2no,!gpiod_get_value(gpio1no));
    return IRQ_HANDLED;
}
irqreturn_t myirq3_handler(int irq,void *dev)
{
    gpiod_set_value(gpio3no,!gpiod_get_value(gpio1no));
    return IRQ_HANDLED;
}
int major;
char kbuf[128]={0};

unsigned int *vrcc;
int mycdev_open(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
ssize_t mycdev_read(struct file *file, char  *ubuf, size_t size, loff_t *lof)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
     unsigned long ret;
    //向用户空间读取拷贝
    if(size>sizeof(kbuf))//用户空间期待读取的大小内核满足不了,那就给内核支持的最大大小
        size=sizeof(kbuf);
    ret=copy_to_user(ubuf,kbuf,size);
    if(ret)//拷贝失败
    {
        printk("copy_to_user filed\n");
        return ret;
    }
    return 0;
}
ssize_t mycdev_write(struct file *file, const char  *ubuf, size_t size, loff_t *lof)
{
    unsigned long ret;
    //从用户空间读取数据
    if(size>sizeof(kbuf))//用户空间期待读取的大小内核满足不了,那就给内核支持的最大大小
        size=sizeof(kbuf);
    ret=copy_from_user(kbuf,ubuf,size);
    if(ret)//拷贝失败
    {
        printk("copy_to_user filed\n");
        return ret;
    }
    switch(kbuf[0])
    {
        case '1':
            if(kbuf[1]=='0')//关灯
            {
                gpiod_set_value(gpio1no,1);
            }
            else
            {
                gpiod_set_value(gpio1no,0);
            }
            break;
        case '2':
            if(kbuf[1]=='0')//关灯
            {
                gpiod_set_value(gpio2no,1);
            }
            else
            {
                gpiod_set_value(gpio2no,0);
            }
            break;
        case '3':
            if(kbuf[1]=='0')//关灯
            {
                gpiod_set_value(gpio3no,1);
            }
            else
            {
                gpiod_set_value(gpio3no,0);
            }
            break;
        
    }
    return 0;
}
int mycdev_close(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
 
//定义操作方法结构体变量并赋值
struct file_operations fops={
 
    .open=mycdev_open,
    .read=mycdev_read,
    .write=mycdev_write,
    .release=mycdev_close,
};

static int __init mycdev_init(void)
{
    major=register_chrdev(0,"mycdev",&fops);
    if(major<0)
    {
        printk("字符设备驱动注册失败\n");
        return major;
    }
    printk("字符设备驱动注册成功:major=%d\n",major);

    int ret1,ret2,ret3;
    dev_led1=of_find_node_by_path("/leds");
    if(dev_led1==NULL)
    {
        printk("设备树解析失败\n");
        return -EFAULT;
    }

    printk("解析设备树成功\n");
    
    dev_irq1=of_find_node_by_path("/myirq");
    if(dev_irq1==NULL)
    {
        printk("设备树解析失败\n");
        return -EFAULT;
    }

    printk("解析设备树成功\n");
    
    //led1
    gpio1no=gpiod_get_from_of_node(dev_led1,"led1-gpios",0,GPIOD_OUT_LOW,NULL);
    if(IS_ERR(gpio1no))
    {
        printk("gpio1对象申请失败\n");
        return -PTR_ERR(gpio1no);
    }
    printk("解析gpio1编号成功\n");

    //led2
    gpio2no=gpiod_get_from_of_node(dev_led1,"led2-gpios",0,GPIOD_OUT_LOW,NULL);
    if(IS_ERR(gpio2no))
    {
        printk("gpio2对象申请失败\n");
        return -PTR_ERR(gpio2no);
    }
    printk("解析gpio编号成功\n");

    //led3
    gpio3no=gpiod_get_from_of_node(dev_led1,"led3-gpios",0,GPIOD_OUT_LOW,NULL);
    if(IS_ERR(gpio3no))
    {
        printk("gpio1对象申请失败\n");
        return -PTR_ERR(gpio3no);
    }
    printk("解析gpio编号成功\n");

    //中断1
    irqno1=irq_of_parse_and_map(dev_irq1,0);
    if(!irqno1)
    {
        printk("软中断解析失败\n");
        return -ENXIO;
    }
    printk("软中断解析成功 irqno1=%d\n",irqno1);
    ret1=request_irq(irqno1,myirq1_handler,IRQF_TRIGGER_FALLING,"key1",NULL);
    if(ret1)
    {
        printk("注册中断失败\n");
        return ret1;
    }
    printk("注册中断成功\n");
    //中断2
    irqno2=irq_of_parse_and_map(dev_irq1,1);
     if(!irqno2)
    {
        printk("软中断解析失败\n");
        return -ENXIO;
    }
    printk("软中断解析成功 irqno2=%d\n",irqno2);
    ret2=request_irq(irqno2,myirq2_handler,IRQF_TRIGGER_FALLING,"key1",NULL);
    if(ret2)
    {
        printk("注册中断失败\n");
        return ret2;
    }
    printk("注册中断成功\n");
    //中断3
    irqno3=irq_of_parse_and_map(dev_irq1,2);
     if(!irqno3)
    {
        printk("软中断解析失败\n");
        return -ENXIO;
    }
    printk("软中断解析成功 irqno3=%d\n",irqno3);
    ret3=request_irq(irqno3,myirq3_handler,IRQF_TRIGGER_FALLING,"key1",NULL);
    if(ret3)
    {
        printk("注册中断失败\n");
        return ret3;
    }
    printk("注册中断成功\n");




    mytimer.expires=jiffies+HZ;
    timer_setup(&mytimer,mytimer_func,0);
    add_timer(&mytimer);
    return 0;
}
static void __exit mycdev_exit(void)
{
    del_timer(&mytimer);
    gpiod_set_value(gpio1no,0);
    gpiod_set_value(gpio2no,0);
    gpiod_set_value(gpio3no,0);
    gpiod_put(gpio1no);
    gpiod_put(gpio2no);
    gpiod_put(gpio3no);

    free_irq(irqno1,NULL);
    free_irq(irqno2,NULL);
    free_irq(irqno3,NULL);

}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

应用

#include
#include
#include 
#include 
#include 
#include
#include
#include "/home/ubuntu/DC22111/char/head.h"
#include 


int main(int argc, char const *argv[])
{
    char buf[128]={0};
    int fd=open("/dev/mycdev",O_RDWR);
    if(fd<0)
    {
        printf("打开设备文件失败\n");
        exit(-1);
    }
    int a,b,c;
    while(1)
    {
        //从终端读取
        printf("请输入1(灯光))>");
        scanf("%d",&a);
        if(a==1)
        {
            printf("请输入功能:1(开灯),2(关灯)");
            scanf("%d",&b);
            if(b==1)
            {
                printf("请输入要操作的灯光1,2,3>");
                scanf("%d",&c);
                ioctl(fd,LED_ON,&c);
            }
            else
            {
                printf("请输入要操作的灯光1,2,3>");
                scanf("%d",&c);
                ioctl(fd,LED_OFF,&c);
            }
        }
        
        
    }

    
    close(fd);

    return 0;
}

你可能感兴趣的:(驱动开发)