成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

手把手教你中斷喚醒系統

系統 Linux
中斷喚醒系統和普通的驅動區別在于,多了兩個函數:suspend 和 resume,在 suspend 函數中,調用 enable_irq_wake,表示該中斷號在系統休眠時也是 enable 狀態,可以觸發中斷。

[[442463]]

在消費類電子中,功耗是很重要的,甚至項目后期一直在調功耗,看看哪里還可以再省電。由此就有了 Linux 電源管理子系統,該子系統包含很多方面:什么時候可以降幀、什么時候可以關掉其他 CPU core、系統運行時如果某外設很少用需要讓它運行時休眠、系統休眠時要保證哪些外設可以喚醒系統。

博主今天要討論的,就是一個按鍵如何喚醒系統,類似于手機的電源鍵。

這個功能并不是新功能,所以 Linux 內部有一個 demo 可以使用,先教大家如何使用該 demo,然后較大家如何撰寫中斷喚醒系統驅動。

官方 demo

demo 目錄:/kernel4.14/drivers/input/keyboard/gpio_keys.c

該驅動是專門為按鍵準備的,是一個身經百戰的驅動,任何時候測試按鍵中斷或者中斷喚醒系統都可以用它,很多時候比自己寫的驅動靠譜。

要想使用該驅動,首先在該目錄的 Makefile 中增加:

  1. obj-y  += gpio_keys.o 

設備樹中增加:

  1. gpio-keys { 
  2.   compatible = "gpio-keys"
  3.   #address-cells = <1>; 
  4.   #size-cells = <0>; 
  5.   autorepeat; 
  6.   key0 { 
  7.    label = "GPIO Key Enter"
  8.    linux,code = <KEY_ENTER>; 
  9.    gpios = <&gpio1 18 GPIO_ACTIVE_LOW>; 
  10.    gpio-key,wakeup; 
  11.   }; 
  12. }; 

compatible 屬性是 “gpio-keys”,gpio_keys.c 文件的674行會匹配這個屬性,匹配到了該驅動就會運行。

linux,code 屬性是按鍵值,Linux 對所有按鍵事件都有編號,所以KEY_ENTER 實際是一個數字,是驅動向上層報告的一個按鍵值。

gpios 屬性是標明哪一個 GPIO 口,低電平觸發,大家可以自己選一個 GPIO。

gpio-key,wakeup 是代表此GPIO支持中斷喚醒,你也可以寫成:wakeup-source。新老版本而已。

修改就是這么簡單,不過語法要符合各位手中的開發板平臺。然后編譯出內核和設備樹文件,下載到板子中。(Linux 內核根目錄會有 .config 文件,確保 CONFIG_PM_SLEEP=y 有打開)

如果驅動加載成功,在 /proc/interrupts 中可以看到:

從左往右第一列是軟件中斷號(唯一)。

第二列是 CPU,表示該中斷在該CPU上觸發了多少次,多核會有多列。

第三列是中斷控制器,imx6ull開發板根中斷控制器是GPC,外部中斷控制器是gpio-mxc,兩者是級聯關系。

第四列是硬件中斷號,也就是GPIO口編號。

第五列表示該中斷是邊沿觸發還是電平觸發。

第六列是中斷名稱,可以找到一個 GPIO Key Enter,如果驅動加載成功就能看到,如果失敗就看不到。

驗證方法

在內核中,休眠方式有很多種,可以通過下面命令查看

  1. # cat /sys/power/state 

常用的休眠方式有freeze、standby、mem、disk

freeze:凍結I/O設備,將它們置于低功耗狀態,使處理器進入空閑狀態,喚醒最快,耗電比其它standby, mem, disk方式高

standby:除了凍結I/O設備外,還會暫停系統,喚醒較快,耗電比其它 mem, disk方式高

mem:將運行狀態數據存到內存,并關閉外設,進入等待模式,喚醒較慢,耗電比disk方式高

disk:將運行狀態數據存到硬盤,然后關機,喚醒最慢

示例:

  1. # echo mem > /sys/power/state 

系統進入睡眠后,基本都會停掉UI、停掉串口,串口無法操作,如圖:

按下按鍵,系統恢復:

當然這里的 log 并不完整,輸入 dmesg 可以看到完整 log:

PM:power manager

具體干了什么,圖中有解釋,分為 suspend 過程和 resume 過程。

其實一個中斷讓它支持喚醒系統,最主要是多了兩個函數:suspend、resume。

suspend 函數在系統整體 suspend 的時候,會調用每個外設注冊的 suspend,我們在這個函數中調用 enable_irq_wake,表示該中斷在系統休眠時是 enable 狀態。

resume 函數在系統整體 resume 的時候,會調用每個外設注冊的 resume 函數,在 resume 函數中調用 disable_irq_wake ,表示該中斷在系統運行時不需要。兩者成對使用。

具體參看下面文章,寫的很好:

http://www.wowotech.net/irq_subsystem/irq_handle_procedure.html

大家也可以研究一下 gpio_keys.c,該驅動看起來比較復雜,但是很完善,畢竟身經百戰,什么因素都考慮到了,測試就用它!

博主寫的 demo

博主下面給的是簡化版,并且自測OK,分享給大家,以后如果需要可以copy

xxx.c

  1. #include <linux/module.h> 
  2. #include <linux/i2c.h> 
  3. #include <linux/interrupt.h> 
  4. #include <linux/delay.h> 
  5. #include <linux/uaccess.h> 
  6. #include <linux/pm.h> 
  7. #include <linux/slab.h> 
  8. #include <linux/sysctl.h> 
  9. #include <linux/proc_fs.h> 
  10. #include <linux/delay.h> 
  11. #include <linux/platform_device.h> 
  12. #include <linux/input.h> 
  13. #include <linux/gpio_keys.h> 
  14. #include <linux/workqueue.h> 
  15. #include <linux/gpio.h> 
  16. #include <linux/of.h> 
  17. #include <linux/of_platform.h> 
  18. #include <linux/of_gpio.h> 
  19. #include <linux/of_irq.h> 
  20. #include <linux/spinlock.h> 
  21. #include <linux/cdev.h> 
  22.  
  23. static int gpionum = 0; 
  24. static int irqnum = 0; 
  25.  
  26. static irqreturn_t my_handler(int irq, void *dev_id) 
  27.  printk("%s\r\n",__FUNCTION__); 
  28.  return IRQ_HANDLED; 
  29.  
  30. static int gpio_keys_probe(struct platform_device *pdev) 
  31.  int ret = 0; 
  32.  struct device_node *node = NULL;; /* 設備節點*/ 
  33.   
  34.  node = of_find_compatible_node(NULL,NULL,"atkalpha-key"); 
  35.  if (node == NULL){ 
  36.   printk("%s:atkalpha-key node not find!\r\n",__FUNCTION__); 
  37.   return -EINVAL; 
  38.  } 
  39.   
  40.  /* 提取 GPIO */ 
  41.  gpionum = of_get_named_gpio(node,"key-gpio", 0); 
  42.  if (gpionum < 0) { 
  43.    printk("of_get_named_gpio can't get key\r\n"); 
  44.  } 
  45.   
  46.  /* 初始化 key 所使用的 IO,并且設置成中斷模式 */ 
  47.  gpio_request(gpionum, "key-gpio"); 
  48.  gpio_direction_input(gpionum);  
  49.   
  50.  irqnum = gpio_to_irq(gpionum); 
  51.   
  52.  ret = request_irq(irqnum,my_handler, IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING, "my-key"NULL); 
  53.  if(ret < 0){ 
  54.   printk("irq %d request failed!\r\n", irqnum); 
  55.   return -EFAULT; 
  56.  } 
  57.  return 0; 
  58.  
  59.  
  60. static const struct of_device_id gpio_keys_of_match[] = { 
  61.  { .compatible = "atkalpha-key", }, 
  62.  { }, 
  63. }; 
  64. MODULE_DEVICE_TABLE(of, gpio_keys_of_match); 
  65.  
  66. static int gpio_keys_remove(struct platform_device *pdev) 
  67.  return 0; 
  68.  
  69. static int gpio_keys_suspend(struct device *dev) 
  70.  printk("%s\r\n",__FUNCTION__); 
  71.  enable_irq_wake(irqnum); 
  72.  return 0; 
  73.  
  74. static int gpio_keys_resume(struct device *dev) 
  75.  printk("%s\r\n",__FUNCTION__); 
  76.  disable_irq_wake(irqnum); 
  77.  return 0; 
  78.  
  79. static SIMPLE_DEV_PM_OPS(gpio_keys_pm_ops, gpio_keys_suspend, gpio_keys_resume); 
  80.  
  81. static struct platform_driver gpio_keys_device_driver = { 
  82.  .probe  = gpio_keys_probe, 
  83.  .remove  = gpio_keys_remove, 
  84.  .driver  = { 
  85.   .name = "my-key"
  86.   .pm = &gpio_keys_pm_ops, 
  87.   .of_match_table = of_match_ptr(gpio_keys_of_match), 
  88.  } 
  89. }; 
  90.  
  91. static int __init gpio_keys_init(void) 
  92.  return platform_driver_register(&gpio_keys_device_driver); 
  93.  
  94. static void __exit gpio_keys_exit(void) 
  95.  platform_driver_unregister(&gpio_keys_device_driver); 
  96.  
  97. module_init(gpio_keys_init); 
  98. module_exit(gpio_keys_exit); 
  99.  
  100. MODULE_LICENSE("GPL"); 
  101. MODULE_AUTHOR("Jason"); 
  102. MODULE_DESCRIPTION("Keyboard driver for GPIOs"); 
  103. MODULE_ALIAS("platform:gpio-keys"); 

xxx.dts

  1. key { 
  2.   #address-cells = <1>; 
  3.   #size-cells = <1>; 
  4.   compatible = "atkalpha-key"
  5.   key-gpio = <&gpio1 18 GPIO_ACTIVE_LOW>; /* KEY0 */ 
  6.   interrupt-parent = <&gpio1>; 
  7.   interrupts = <18 IRQ_TYPE_EDGE_BOTH>; /* FALLING RISING */ 
  8.   gpio-key,wakeup; 
  9.   status = "okay"
  10. }; 

最后再總結一下:中斷喚醒系統和普通的驅動區別在于,多了兩個函數:suspend 和 resume,在 suspend 函數中,調用 enable_irq_wake,表示該中斷號在系統休眠時也是 enable 狀態,可以觸發中斷。在 resume 函數中,調用 disable_irq_wake ,恢復原始的中斷觸發路徑。

然后使用 SIMPLE_DEV_PM_OPS 宏將 suspend 和 resume 函數注冊到 gpio_keys_pm_ops 操作集,最終由 platform 注冊到系統中。這樣完成后,系統休眠過程中就會調用到設備注冊的 suspend,系統喚醒過程中就會調用設備注冊的 resume 函數。

至于 probe 函數的書寫,我在 GPIO 子系統和中斷子系統系列文章都講過這些函數的使用,大家可以去我的網站查看:

http://www.linuxer.vip

note:該 demo 只用來喚醒系統,如果你的中斷是在 I2C 等設備驅動中,喚醒系統后要立刻在中斷處理函數中進行 I2C 通信,寫法不太一樣,但是框架相同。

另外,該驅動的中斷處理函數中沒做什么東西,因此喚醒后執行完中斷處理函數后又會睡過去。如果你想要該中斷喚醒系統后讓系統一直處于喚醒狀態,請在中斷處理函數中使用 __pm_stay_awake() 和 __pm_relax()函數。

 

責任編輯:姜華 來源: 嵌入式Linux系統開發
相關推薦

2022-01-08 20:04:20

攔截系統調用

2011-01-10 14:41:26

2025-05-07 00:31:30

2011-05-03 15:59:00

黑盒打印機

2021-07-14 09:00:00

JavaFX開發應用

2021-12-15 08:49:21

gpio 子系統pinctrl 子系統API

2011-02-22 13:46:27

微軟SQL.NET

2021-02-26 11:54:38

MyBatis 插件接口

2023-04-26 12:46:43

DockerSpringKubernetes

2022-03-14 14:47:21

HarmonyOS操作系統鴻蒙

2022-07-27 08:16:22

搜索引擎Lucene

2022-12-07 08:42:35

2009-07-19 15:02:56

2021-09-22 08:51:34

Android

2020-07-09 08:59:52

if else模板Service

2020-04-14 10:20:12

MySQL數據庫死鎖

2021-08-04 08:55:02

Socket Java開發

2011-01-06 10:39:25

.NET程序打包

2016-04-27 09:49:16

用戶模型產品總結

2021-09-30 18:27:38

數據倉庫ETL
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 天堂久久一区 | 国产亚洲精品精品国产亚洲综合 | 久久亚洲国产 | 妞干网av| 99视频在线免费观看 | 成人免费视频 | 亚洲精品一区二区 | 欧美成人aaa级毛片在线视频 | 亚洲第一区久久 | 欧美xxxx黑人又粗又长 | 一区二区精品 | 一区二区三区精品在线视频 | 天天插天天操 | 夜夜骑天天干 | 一区二区三区四区五区在线视频 | 青青久久 | 九九热在线免费视频 | 国产精品久久久久久久免费观看 | 视频一区中文字幕 | 久久精品久久久 | 国产精品不卡一区 | 亚洲毛片在线观看 | 成人av一区二区亚洲精 | 美女国内精品自产拍在线播放 | 欧美性生交大片免费 | 国产成人精品久久二区二区91 | 电影午夜精品一区二区三区 | 完全免费av在线 | 一级高清视频 | 天天干天天爱天天 | 青草青草久热精品视频在线观看 | 999精品视频 | 国产美女在线观看 | 久久精品男人的天堂 | 亚洲乱码一区二区 | 国内自拍偷拍一区 | 黑人一级片视频 | 国产精品视频一 | 欧美一区在线视频 | 亚洲精品视频一区 | 国产精品一区二区久久 |