1.使用GPIO子系統,編寫LED驅動,應用程序測試
mychrdev.c
#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/io.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/gpio.h>
int major; // 保存申請的主設備號
char kbuf[128]={0};struct gpio_desc *gpiono1;
struct gpio_desc *gpiono2;
struct gpio_desc *gpiono3;
struct device_node *dnode;//保存解析到的設備樹節點地址
// 封裝常用的操作方法
int mychrdev_open(struct inode *inode, struct file *file)
{printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);return 0;
}
ssize_t mychrdev_read(struct file *file, char __user *ubuf, size_t size, loff_t *lof)
{printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);unsigned int ret;ret=copy_to_user(ubuf,kbuf,size);if(ret){printk("copy_to_user err\n");return -ret;}return 0;
}
ssize_t mychrdev_write(struct file *file, const char __user *ubuf, size_t size, loff_t *lof)
{//ubuf就是應用程序中write函數第二個參數傳過來的值//size就是應用程序中write函數第三個參數傳過來的值printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);unsigned int ret;ret=copy_from_user(kbuf,ubuf,size);if(ret){printk("copy_from_user err\n");return -ret;}if(kbuf[0]=='0')//關燈{gpiod_set_value(gpiono1,0);//輸出低電平gpiod_set_value(gpiono2,0);//輸出低電平gpiod_set_value(gpiono3,0);//輸出低電平}else if(kbuf[0]=='1')//開led1燈{gpiod_set_value(gpiono1,1);//輸出高電平}else if(kbuf[0]=='2')//開led2燈{gpiod_set_value(gpiono2,1);//輸出高電平}else if(kbuf[0]=='3')//開led3燈{gpiod_set_value(gpiono3,1);//輸出高電平}return 0;
}
int mychrdev_close(struct inode *inode, struct file *file)
{printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);return 0;
}
struct file_operations fops = {.open=mychrdev_open,.release=mychrdev_close,.read=mychrdev_read,.write=mychrdev_write,
};
static int __init mycdev_init(void)
{// 注冊字符設備驅動major = register_chrdev(0, "mychrdev", &fops);if (major < 0){printk("注冊字符設備驅動失敗%d\n", __LINE__);}printk("注冊字符設備驅動成功 major=%d\n", major);//解析LED燈設備樹節點dnode=of_find_node_by_path("/leds");if(dnode==NULL){printk("解析設備樹節點失敗\n");return -ENXIO;}printk("解析設備樹節點成功\n");//解析LED1的gpio編號gpiono1=gpiod_get_from_of_node(dnode,"led1-gpio",0, GPIOD_OUT_LOW,NULL);if(gpiono1==NULL){printk("解析gpio編號失敗\n");return -ENXIO;}printk("解析gpio編號成功\n");gpiono2=gpiod_get_from_of_node(dnode,"led2-gpio",0, GPIOD_OUT_LOW,NULL);if(gpiono2==NULL){printk("解析gpio編號失敗\n");return -ENXIO;}printk("解析gpio編號成功\n");gpiono3=gpiod_get_from_of_node(dnode,"led3-gpio",0, GPIOD_OUT_LOW,NULL);if(gpiono3==NULL){printk("解析gpio編號失敗\n");return -ENXIO;}printk("解析gpio編號成功\n");return 0;
}
static void __exit mycdev_exit(void)
{//釋放gpio編號gpiod_put(gpiono1);gpiod_put(gpiono2);gpiod_put(gpiono3);// 注銷字符設備驅動unregister_chrdev(major, "mychrdev");
}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");
led.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
int main(int argc, char const *argv[])
{char buf[128] = {};int fd = open("/dev/mychrdev", O_RDWR);if (fd < 0){printf("打開設備文件失敗\n");return -1;}while (1){printf("請輸入控制碼:1(led1開燈) 2(led2開燈) 3(led3開燈) 0(關燈)>");fgets(buf, sizeof(buf), stdin); // 從終端輸入一個數據buf[strlen(buf) - 1] = '\0';write(fd, buf, sizeof(buf));}close(fd);return 0;
}
2.注冊三個按鍵的中斷,只需要寫內核代碼
myled.c
#include <linux/init.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/gpio.h>
#include <linux/timer.h>
#include <linux/interrupt.h>
#include <linux/of_irq.h>
/*myirq{compatible = "hqyj,myirq";interrupt-parent = <&gpiof>;interrupts=<9 0>,<7 0>,<8 0>;
};
*/struct device_node *dnode_led; // 保存解析到的led設備樹節點地址
struct device_node *dnode_int; // 保存解析到的中斷設備樹節點地址
unsigned int irqno1; // 按鍵1軟中斷號
unsigned int irqno2; // 按鍵2軟中斷號
unsigned int irqno3; // 按鍵3軟中斷號
struct gpio_desc *gpiono1; // led1設備號
struct gpio_desc *gpiono2; // led2設備號
struct gpio_desc *gpiono3; // led3設備號// 中斷處理函數1
irqreturn_t myirq_handler1(int irq, void *dev)
{printk("key1_intc\n");//關燈三 開燈一gpiod_set_value(gpiono3, 0);gpiod_set_value(gpiono1, 1);return IRQ_HANDLED;
}
// 中斷處理函數2
irqreturn_t myirq_handler2(int irq, void *dev)
{printk("key2_intc\n");//關燈一 開燈二gpiod_set_value(gpiono1, 0);gpiod_set_value(gpiono2, 1);return IRQ_HANDLED;
}
// 中斷處理函數3
irqreturn_t myirq_handler3(int irq, void *dev)
{printk("key3_intc\n");//關燈二 開燈三gpiod_set_value(gpiono2, 0);gpiod_set_value(gpiono3, 1);return IRQ_HANDLED;
}static int __init mycdev_init(void)
{// 解析按鍵的設備樹節點dnode_int = of_find_compatible_node(NULL, NULL, "hqyj,myirq");if (dnode_int == NULL){printk("解析設備樹節點失敗\n");return -ENXIO;}printk("解析設備樹節點成功\n");// 解析LED燈設備樹節點dnode_led = of_find_node_by_path("/leds");if (dnode_led == NULL){printk("解析設備樹節點失敗\n");return -ENXIO;}printk("解析設備樹節點成功\n");// 解析按鍵1的軟中斷號irqno1 = irq_of_parse_and_map(dnode_int, 0);if (!irqno1){printk("解析按鍵1軟中斷號失敗\n");return -ENXIO;}printk("解析按鍵1軟中斷號成功%d\n", irqno1);// 解析按鍵2的軟中斷號irqno2 = irq_of_parse_and_map(dnode_int, 1);if (!irqno2){printk("解析按鍵2軟中斷號失敗\n");return -ENXIO;}printk("解析按鍵2軟中斷號成功%d\n", irqno2);// 解析按鍵3的軟中斷號irqno3 = irq_of_parse_and_map(dnode_int, 2);if (!irqno3){printk("解析按鍵3軟中斷號失敗\n");return -ENXIO;}printk("解析按鍵3軟中斷號成功%d\n", irqno3);// 注冊按鍵1中斷int ret1 = request_irq(irqno1, myirq_handler1, IRQF_TRIGGER_FALLING, "key1", (void *)1);if (ret1){printk("中斷注冊失敗\n");return ret1;}printk("中斷注冊成功\n");// 注冊按鍵2中斷int ret2 = request_irq(irqno2, myirq_handler2, IRQF_TRIGGER_FALLING, "key2", (void *)2);if (ret2){printk("中斷注冊失敗\n");return ret2;}printk("中斷注冊成功\n");// 注冊按鍵3中斷int ret3 = request_irq(irqno3, myirq_handler3, IRQF_TRIGGER_FALLING, "key3", (void *)3);if (ret3){printk("中斷注冊失敗\n");return ret3;}printk("中斷注冊成功\n");// 解析LED1的gpio編號gpiono1 = gpiod_get_from_of_node(dnode_led, "led1-gpio", 0, GPIOD_OUT_LOW, NULL);if (gpiono1 == NULL){printk("解析gpio編號失敗\n");return -ENXIO;}printk("解析gpio編號成功\n");// 解析LED2的gpio編號gpiono2 = gpiod_get_from_of_node(dnode_led, "led2-gpio", 0, GPIOD_OUT_LOW, NULL);if (gpiono2 == NULL){printk("解析gpio編號失敗\n");return -ENXIO;}printk("解析gpio編號成功\n");// 解析LED3的gpio編號gpiono3 = gpiod_get_from_of_node(dnode_led, "led3-gpio", 0, GPIOD_OUT_LOW, NULL);if (gpiono3 == NULL){printk("解析gpio編號失敗\n");return -ENXIO;}printk("解析gpio編號成功\n");return 0;
}
static void __exit mycdev_exit(void)
{// 釋放軟中斷號free_irq(irqno1, (void *)1);free_irq(irqno2, (void *)2);free_irq(irqno3, (void *)3);// 釋放gpio編號gpiod_put(gpiono1);gpiod_put(gpiono2);gpiod_put(gpiono3);
}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");