2月22日作業,按鍵中斷LED燈控制

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");

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/696477.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/696477.shtml
英文地址,請注明出處:http://en.pswp.cn/news/696477.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

微軟Azure OpenAI的 GPT 接口使用小結

直接使用OpenAI的 GPT服務&#xff0c;在國內環境使用上會一些相關問題&#xff0c;微軟提供了OpenAI的服務&#xff0c;基本上可以滿足的相關的需要。下面提供一些簡單的使用操作&#xff0c;來讓你快速使用到 GPT 的服務。 前提&#xff1a;注冊Azure的賬戶&#xff0c;并綁…

OpenCV中的normalize函數以及NORM_MINMAX、NORM_INF、NORM_L1、NORM_L2具體應用介紹

在OpenCV中&#xff0c;normalize函數用于將圖像或矩陣的值規范化到一個特定的范圍內。這在圖像處理中非常有用&#xff0c;比如在調整圖像的對比度、準備數據進行機器學習處理時。規范化可以提高不同圖像之間的可比性&#xff0c;或是為了滿足特定算法對數據范圍的要求。 nor…

數的反碼和補碼表示

2.反碼 反碼的表示方法是: 正數的反碼是其本身負數的反碼是在其原碼的基礎上,符號位不變&#xff0c;其余各個位取反 [1][000000011原[000000011反[-1][10000001]原[11111110]反 3.補碼 補碼的表示方法是: 正數的補碼就是其本身 負數的補碼是在其原碼的基礎上,符號位不變,其余各…

36、IO進程線程/進程和線程之間的通信練習

一、使用有名管道完成兩個進程的相互通信(提示&#xff1a;可以使用多進程或多線程完成)。 代碼1&#xff1a;創建兩個有名管道文件 #include<myhead.h>int main(int argc, const char *argv[]) {if(mkfifo("./mingtohua",0664)-1)//創建小明向小華發信息的管…

Stable Diffusion 繪畫入門教程(webui)-ControlNet(深度Depth)

上篇文章介紹了線稿約束&#xff0c;這篇文章介紹下深度Depth 文章目錄 一、選大模型二、寫提示詞三、基礎參數設置四、啟用ControlNet 顧名思義&#xff0c;就是把原圖預處理為深度圖&#xff0c;而深度圖可以區分出圖像中各元素的遠近關系&#xff0c;那么啥事深度圖&#xf…

c/c++ | 字符串函數總結 | 為什么總喜歡糾結sizeof 和strlen 呢?

其實時間長了&#xff0c;稍微研究后&#xff0c;再來品味&#xff0c;別有一番滋味 總是看著混亂&#xff0c;但是靜下來看&#xff0c;還是能琢磨透的&#xff0c;只是看著復雜&#xff0c;本質是兩套風格&#xff0c;然后又要有交集&#xff0c;所以就看起來復雜 // 首先字符…

目標管理SMART原則

SMART原則是一種目標管理方法&#xff0c;它包括以下五個要素&#xff1a; 具體性&#xff08;Specific&#xff09;&#xff1a;目標應該是明確的&#xff0c;具體地說明要達成的行為標準。例如&#xff0c;一個目標可能描述為“減少客戶投訴率”&#xff0c;而不是“增強客戶…

本機防攻擊簡介

定義 在網絡中&#xff0c;存在著大量針對CPU&#xff08;Central Processing Unit&#xff09;的惡意攻擊報文以及需要正常上送CPU的各類報文。針對CPU的惡意攻擊報文會導致CPU長時間繁忙的處理攻擊報文&#xff0c;從而引發其他業務的中斷甚至系統的中斷&#xff1b;大量正常…

惠爾頓 網絡安全審計系統 任意文件讀取漏洞復現

0x01 產品簡介 惠爾頓網絡安全審計產品致力于滿足軍工四證、軍工保密室建設、國家涉密網絡建設的審計要求&#xff0c;規范網絡行為&#xff0c;滿足國家的規范&#xff1b;支持1-3線路的internet接入、1-3對網橋&#xff1b;含強大的上網行為管理、審計、監控模塊&#xff1b…

【2024軟件測試面試必會技能】Requests(5):Requests模塊_設置代理

設置代理 代理&#xff08;英語&#xff1a;Proxy&#xff09;&#xff0c;也稱網絡代理&#xff0c;是一種特殊的網絡服務&#xff0c;英文全稱是&#xff08;Proxy Server&#xff09;&#xff0c;其功 能就是代理網絡用戶去取得網絡信息。形象的說&#xff1a;它是網絡信息…

正向代理和反向代理釋義

代理 客戶端 代理 服務端 對客戶端而言&#xff0c;代理是服務端&#xff1b;對服務端而言&#xff0c;代理是客戶端。這個很好理解吧&#xff0c;以祖孫三代關系為例&#xff0c;爸爸在兒子面前是爸爸&#xff0c;爸爸在爺爺面前是兒子。 無論是正向代理還是反向代理&#…

Android14 InputManager-ANR原理

目標窗口查找時&#xff0c;作為派發目標的窗口必須已經準備好接收新的輸入事件&#xff0c;否則判定窗口處于未響應狀態&#xff0c;終止事件的派發過程&#xff0c;并在一段時間后再試。倘若5s后窗口仍然未準備好接收輸入事件&#xff0c;將導致ANR。直接引發ANR的原因有很多…

操作系統學習記錄

系統篇 內核 應用和底層硬件&#xff08;CPU、內存、硬盤等&#xff09;的連接橋梁。 用戶態和內核態 CPU和進程可以在兩種態下運行。 內核態可以直接訪問所有硬件資源&#xff0c;用戶態需要通過“系統調用”陷入到內核態才能否則只能訪問用戶空間地址的內存&#xff08;虛…

P8630 [藍橋杯 2015 國 B] 密文搜索

P8630 [藍橋杯 2015 國 B] 密文搜索 - 洛谷 | 計算機科學教育新生態 (luogu.com.cn)https://www.luogu.com.cn/problem/P8630 題目分析 基本上是hash的板子&#xff0c;但實際上對于密碼串&#xff0c;只要判斷主串中任意連續的八個位置是否存在密碼串即可&#xff1b;那么我們…

PHP學習筆記1——html標簽以及頭部元素頁面布局

html是一種超文本標識符號&#xff0c;用來在網頁中指定顯示頁面格式顯示 基本格式 <!doctype html> <html><head><title></title> </head><body> </body></html> 包含聲明&#xff0c;框架html&#xff0c;頭部head&a…

怎么把公眾號文章鏈接做成二維碼?掃碼查看公眾號推文的方法

公眾號是現在給用戶分享內容的一種方式&#xff0c;通過輸出優質的公眾號文章內容來為關注者提供信息和內容。當我們發布公眾號文章后&#xff0c;有些情況下會需要將公眾號內容生成二維碼之后&#xff0c;印刷到傳單、展板上來顯示&#xff0c;那么如何將公眾號的文章鏈接轉二…

Commonjs 和 Es Module詳解

一 前言 今天我們來深度分析一下 Commonjs 和 Es Module&#xff0c;希望通過本文的學習&#xff0c;能夠讓大家徹底明白 Commonjs 和 Es Module 原理&#xff0c;能夠一次性搞定面試中遇到的大部分有關 Commonjs 和 Es Module 的問題。 帶上疑問開始今天的分析&#xff1a; …

數字化轉型導師堅鵬:城市數字化轉型頂層規劃方法

城市數字化轉型頂層規劃方法 課程背景&#xff1a; 很多城市存在以下問題&#xff1a; 不知道如何系統地開展數字化轉型工作&#xff1f; 不清楚如何科學地制定數字化轉型戰略&#xff1f; 不知道如何高效地實施數字化轉型戰略&#xff1f; 課程特色&#xff1a; 有…

基于SpringBoot的在線拍賣系統設計與實現(源碼+調試+LW+PPT)

項目描述 臨近學期結束&#xff0c;還是畢業設計&#xff0c;你還在做java程序網絡編程&#xff0c;期末作業&#xff0c;老師的作業要求覺得大了嗎?不知道畢業設計該怎么辦?網頁功能的數量是否太多?沒有合適的類型或系統?等等。今天給大家介紹一篇基于SpringBoot的在線拍…

C++編程知識

&#xff08;1&#xff09;把數組中的所有元素賦值為一個值 memset(arr,0x3f,sizeof(arr))//最大值 memset(arr,0,sizeof(distance));//賦值為0&#xff08;2&#xff09;找最大最小值的下標 int zuixiao-1//設置dist-1只是因為方便選取第一個數作為比較對象而已 for(i0;i<…