驱动——按键中断控制LED灯实验

这篇具有很好参考价值的文章主要介绍了驱动——按键中断控制LED灯实验。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

三个按键实现按键中断

要求:按键按一下灯亮,再按一下灯灭

注:由于开发板位置,为了方便一一对应观察,采用key1控制LED3,key2控制LED2,key3控制LED1

0、添加相关设备树节点信息

①按键相关引脚,通过原理图可得:

key1------>gpiof9;

key2------>gpiof7;

key3------>gpiof8

②LED灯相关引脚:

led1----->gpioe10;

led2------>gpiof10;

led3------>gpioe8;

③编写完成之后在内核顶层目录下执行make dtbs重新生成设备树二进制文件

cp arch/arm/boot/dts/stm32mp157a-fsmp1a.dtb ~/tftpboot

④ 重新启动开发板

  myleds{
      myled1 = <&gpioe 10 0>;
      myled2 = <&gpiof 10 0>;
      myled3 = <&gpioe 8 0>;
  };

  myirq{
      interrupt-parent = <&gpiof>;
      interrupts = <7 0>,<8 0>,<9 0>;
  };

1、获取设备树节点

本次采用通过名字进行获取(具体详情见gpio子系统(LED灯的操控实验))

2、从节点信息中解析出软中断号

int irq_of_parse_and_map(struct device_node *dev, int index)

功能:解析设备树节点信息获取软中断号

参数:

dev:设备树节点信息结构体指针

index:中断的索引号

返回值:成功返回软中断号,失败返回0

 3、将要使用的中断注册进内核

int request_irq(unsigned int irq, irq_handler_t handler, unsigned long flags, const char *name, void *dev)

功能:注册中断

参数:

        irq:软中断号

        handler:中断处理函数 //typedef irqreturn_t (*irq_handler_t)(int, void *);

                //中断处理函数返回值:

                        //IRQ_NONE //中断不是设备处理或者没处理

                        //IRQ_HANDLED //中断正常处理

                        //IRQ_WAKE_THREAD //开启中断处理线程

        flags:中断触发的方式

                IRQF_TRIGGER_RISING//上升沿触发

                IRQF_TRIGGER_FALLING//下降沿触发

                IRQF_TRIGGER_HIGH//高电平触发

                IRQF_TRIGGER_LOW//低电平触发

        name:中断的名字//自己指定

                sudo cat /proc/interrupts

        dev:在这个函数里传给中断处理函数的参数,根据情况决定给是否使用

返回值:成功返回0,失败返回错误码

 4、注销中断

void *free_irq(unsigned int irq, void *dev_id)

功能:注销中断

参数:

        irq:软中断号 d

        ev_id:向中断处理函数中传递的参数

返回值:返回设备名

 5、具体代码实现

#include<linux/init.h>
#include<linux/module.h>
#include<linux/of.h>
#include<linux/of_gpio.h>
#include<linux/gpio.h>
#include<linux/fs.h>
#include<linux/uaccess.h>
#include<linux/io.h>
#include<linux/cdev.h>
#include<linux/slab.h>
#include<linux/timer.h>
#include<linux/of_irq.h>
#include<linux/interrupt.h>
#include"./six.h"

#define GNAME "mydev"


unsigned int irqno1;//用于接收软中断号
unsigned int irqno2;//用于接收软中断号
unsigned int irqno3;//用于接收软中断号
unsigned int num=0;

//定义一个底半部
struct tasklet_struct tasklet;

irqreturn_t irq_handler1(int irqno,void *dev)
{
    num++;
    printk("key1 interrupt %d ...\n",num);
    gpiod_set_value(gpiono1,(!gpiod_get_value(gpiono1)));
    return IRQ_HANDLED;
}
irqreturn_t irq_handler2(int irqno,void *dev)
{
    num++;
    printk("key2 interrupt %d ...\n",num);
    gpiod_set_value(gpiono2,(!gpiod_get_value(gpiono2)));
    return IRQ_HANDLED;
}
irqreturn_t irq_handler3(int irqno,void *dev)
{
    num++;
    printk("key3 interrupt %d ...\n",num);
    gpiod_set_value(gpiono3,(!gpiod_get_value(gpiono3)));
    return IRQ_HANDLED;
}
static int __init mynode_init(void)
{
   
    int ret;
    //通过名字获取设备树节点信息
    node = of_find_node_by_name(NULL,"myirq");
    if(NULL == node)
    {
        printk("of find node by name error\n");
        return -EFAULT;
    }
    //根据设备树节点获取软中断号1
    irqno1 = irq_of_parse_and_map(node,1);
    if(irqno1==0)
    {
        printk("irq of parse and map error\n");
        return EINVAL;
    }
    printk("irq of parse and map success\n");
    //根据设备树节点获取软中断号2
    irqno2 = irq_of_parse_and_map(node,0);
    if(irqno2==0)
    {
        printk("irq of parse and map error\n");
        return EINVAL;
    }
    printk("irq of parse and map success\n");
    //根据设备树节点获取软中断号3
    irqno3 = irq_of_parse_and_map(node,2);
    if(irqno3==0)
    {
        printk("irq of parse and map error\n");
        return EINVAL;
    }
    printk("irq of parse and map success\n");
    //将要使用的中断注册进内核
    ret = request_irq(irqno1,irq_handler1,IRQF_TRIGGER_FALLING,"key",NULL);
    if(ret)
    {
        printk("request irq irqno1 error\n");
        return ret;
    }
    //将要使用的中断注册进内核
    ret = request_irq(irqno2,irq_handler2,IRQF_TRIGGER_FALLING,"key",NULL);
    if(ret)
    {
        printk("request irq irqno1 error\n");
        return ret;
    }
    //将要使用的中断注册进内核
    ret = request_irq(irqno3,irq_handler3,IRQF_TRIGGER_FALLING,"key",NULL);
    if(ret)
    {
        printk("request irq irqno1 error\n");
        return ret;
    }
     //通过名字获取设备树节点信息
    node = of_find_node_by_name(NULL,"myleds");
    if(NULL == node)
    {
        printk("of find node by name error\n");
        return -EFAULT;
    }
    //获取并申请LED1的gpio编号
    gpiono1 = gpiod_get_from_of_node(node,"myled1",0,GPIOD_OUT_LOW,NULL);
    if(IS_ERR(gpiono1))
    {
        printk("1gpiod get from of node error\n");
        return PTR_ERR(gpiono1);
    }
    //获取并申请LED2的gpio编号
     gpiono2 = gpiod_get_from_of_node(node,"myled2",0,GPIOD_OUT_LOW,NULL);
    if(IS_ERR(gpiono2))
    {
        printk("2gpiod get from of node error\n");
        return PTR_ERR(gpiono2);
    }
    //获取并申请LED3的gpio编号
     gpiono3 = gpiod_get_from_of_node(node,"myled3",0,GPIOD_OUT_LOW,NULL);
    if(IS_ERR(gpiono3))
    {
        printk("3gpiod get from of node error\n");
        return PTR_ERR(gpiono3);
    }
    
    return 0;
    
}

static void __exit mynode_exit(void)
{
   
    //注销中断1
    free_irq(irqno1,NULL);
    //注销中断1
    free_irq(irqno2,NULL);
    //注销中断1
    free_irq(irqno3,NULL);
    //卸载驱动前熄灭灯LED1
    gpiod_set_value(gpiono1,0);
        //卸载驱动前熄灭灯LED1
    gpiod_set_value(gpiono2,0);
        //卸载驱动前熄灭灯LED1
    gpiod_set_value(gpiono3,0);
        
    //释放申请得到的LED1gpio编号
    gpiod_put(gpiono1);
        //释放申请得到的LED2gpio编号
    gpiod_put(gpiono2);
        //释放申请得到的LED3gpio编号
    gpiod_put(gpiono3);
        

}

module_init(mynode_init);
module_exit(mynode_exit);

MODULE_LICENSE("GPL");

 头文件代码:

#ifndef __LED_H__
#define __LED_H__

#define LED_ON _IOW('a',1,int)
#define LED_OFF _IOW('a',0,int)


struct device_node *node;
struct gpio_desc *gpiono1;
struct gpio_desc *gpiono2;
struct gpio_desc *gpiono3;
int ret;

typedef enum{
    LED1,
    LED2,
    LED3,  
}led_t;

#endif

makefile脚本文件见gpio子系统(LED灯的操控实验) 

6、实验现象

见按键中断控制LED灯实验现象文章来源地址https://www.toymoban.com/news/detail-416837.html

到了这里,关于驱动——按键中断控制LED灯实验的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 2.外部中断方式检测按键输入并控制LED示例

    打断CPU执行正常的程序,转而处理紧急程序,然后返回原暂停的程序继续运行,就叫中断。 其中,系统中断表 及对应的说明 如下图所示: 我们可以按住 Ctrl 点击中断向量表中的 ISR 函数名,可以进入该函数,可以发现,这些系统函数虽然都定义了,但是没有编写其工作方式

    2024年04月13日
    浏览(42)
  • ARM day9 (按键中断控制led亮灭)

     保存现场 异常的处理流程,保存现场会做哪些事情(四大步三小步) ----CPU自动完成 1.保存CPSR寄存器中的值,到SPSR_mode寄存器中 2.修改CPSR寄存器对应的位         1修改CPSR寄存器对应的状态位(T位)         2根据需要,禁止相应中断位(I位 / F位)         3修改CPSR寄存

    2024年02月13日
    浏览(43)
  • STM32——05-按键、时钟控制、中断复位 点亮LED灯

    如何点亮一颗LED灯   编程实现点灯 常用的 GPIO HAL 库函数: void HAL_GPIO_Init ( GPIO_TypeDef * GPIOx , GPIO_InitTypeDef * GPIO_Init ); void HAL_GPIO_WritePin ( GPIO_TypeDef * GPIOx , uint16_t GPIO_Pin , GPIO_PinState PinState ); void HAL_GPIO_TogglePin ( GPIO_TypeDef * GPIOx , uint16_t GPIO_Pin );  结构体  GPIO_InitTypeDef  定义:

    2024年02月08日
    浏览(45)
  • 【Linux驱动】设备树中指定中断 | 驱动中获得中断 | 按键中断实验

    🐱作者:一只大喵咪1201 🐱专栏:《Linux驱动》 🔥格言: 你只管努力,剩下的交给时间! 继续拿这个中断流程图来说话。 在硬件上,中断控制器只有 GIC 这一个,但是我们在软件上可以把 GPIO 也归类为中断控制器。 芯片会有多个GPIO模块,所以软件上的中断控制器就会有很

    2024年01月17日
    浏览(51)
  • 按键控制LED闪烁实验

    实验任务: 本节实验任务是使用底板上的PL_KEY0和PL_KEY1按键来控制底板上的PL_LED0和PL_LED1两个LED的闪烁方式。没有按键按下时,两个LED保持常亮;如果按键0按下,则两个LED交替闪烁;如果按键1按下,则两个LED同时闪烁。 模块示意图:   模块代码实现: modulekey_led( input     

    2024年02月04日
    浏览(43)
  • 按键控制 LED 实验

    目录 1 按键简介 2 实验任务 3 硬件设计 4 程序设计 5 下载验证         按键开关是一种电子开关,属于电子元器件类。         轻触式按键         自锁式按键         使用底板上的 PL_KEY0 和 PL_KEY1 按键来控制底板上的 PL_LED0 和 PL_LED1 两个 LED 的闪烁方式。没有

    2024年02月11日
    浏览(38)
  • 51单片机实验3-中断控制的LED灯(单一外部中断控制LED灯,两个外部中断控制LED灯,交通管制路口)

    1、编写程序通过单一外部中断控制LED灯,每按一次开关,中断服务程序中,低4位led和高4位led交替闪烁5次; 2、编写程序利用两个外部中断控制LED灯。上电后8个led以流水灯方式点亮间隔100ms,K1按下,流水灯反向显示1次间隔50ms;按下K2,8个led同时闪烁3次,间隔50ms。 电路连接

    2024年02月08日
    浏览(47)
  • 单片机实验三 独立按键控制LED实验

    1.熟悉实验软件和硬件,进行正确的接线; 2.8个不同的按键对应不同的8个灯,按下按键对应的灯就会亮。 1.Keil uVision4 2.PZISP自动下载软件 3.HC6800S开发板 按键实物图 按键按下就会产生一个脉冲信号,检测脉冲信号就可以知道是否有按键被按下去。 按键原理图 按键之间相互独

    2024年02月08日
    浏览(40)
  • STM32G030C8T6:使用按键控制LED亮灭(外部中断)

    本专栏记录STM32开发各个功能的详细过程,方便自己后续查看,当然也供正在入门STM32单片机的兄弟们参考; 本小节的目标是,系统主频64 MHZ,采用高速外部晶振,通过KEY1 按键的PA0 引脚配置成中断输入引脚,PB9引脚配置成输出,每次按键,PA0 引脚就会进入一次外部中断,每进

    2024年02月05日
    浏览(40)
  • 基于STM32CubeMX与keil采用按键外部中断方式控制LED与蜂鸣器

    这篇文章详细记录外部中断方式控制LED的亮灭以及蜂鸣器的开关;本文从原理图开始到最后实现功能,内容详细。 本栏目的所有都是基于STM32F407ZET6芯片,博主采用的是普中的天马F407开发板。 实现功能:LED0与LED1默认熄灭,蜂鸣器默认关闭。按下按键KEY0,控LED0亮灭;按下按

    2024年02月16日
    浏览(48)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包