0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

i.MX6ULL學(xué)習(xí)筆記:GPIO按鍵輸入捕獲

玩轉(zhuǎn)單片機 ? 來源:玩轉(zhuǎn)單片機 ? 2023-04-06 10:37 ? 次閱讀

掌握GPIO的輸入輸出是必備技能,這篇推文繼續(xù)來學(xué)習(xí)操作GPIO,這篇主要學(xué)習(xí)GPIO的輸入,通過使用按鍵進行GPIO輸入實驗。

|硬件

寫驅(qū)動前先看看具體的硬件,資料來源于野火公開文件,更多資料請自行訪問野火官網(wǎng)查詢:

68720836-d329-11ed-bfe3-dac502259ad0.png

開發(fā)板上的四個按鍵功能如下圖所示,選擇KEY按鍵來進行按鍵輸入實驗:

6887674e-d329-11ed-bfe3-dac502259ad0.png

|驅(qū)動

1、準備驅(qū)動文件:

68aae52a-d329-11ed-bfe3-dac502259ad0.png

2、遠程連接ubuntu并打開設(shè)備樹和驅(qū)動文件夾:

68d1a160-d329-11ed-bfe3-dac502259ad0.png

3、在設(shè)備樹中添加 pinctrl 節(jié)點

GPIO的相關(guān)配置可以在arch/arm/boot/dts/imx6ul-pinfunc.h文件中找到對應(yīng)的宏定義:

69098332-d329-11ed-bfe3-dac502259ad0.png

在設(shè)備樹中添加節(jié)點:

pinctrl_key: keygrp {
  fsl,pins = <
    MX6UL_PAD_SNVS_TAMPER1__GPIO5_IO01 0xF080 /* key */ 
  >;
};

相關(guān)解析:

/*PAD寄存器的配置說明*/
0xF080,即1000 0000 1000 0000


/*
*bit 16:0 HYS關(guān)閉
*bit [15:14]: [00]下拉 [01]47k上拉 [10]100k上拉 [11]22k上拉 <---
*bit [13]: [0]kepper功能 [1]pull功能
*bit [12]: [0]pull/keeper-disable [1]pull/keeper-enable
*bit [11]: 0 關(guān)閉開路輸出
*bit [10:8]: 00 保留值
*bit [7:6]: 10 速度100Mhz
*bit [5:3]: 000 輸出disable <---
*bit [2:1]: 00 保留值
*bit [0]: 0 低轉(zhuǎn)換率
*/

4、在設(shè)備樹中添加KEY設(shè)備節(jié)點

key {
  #address-cells = <1>;
  #size-cells = <1>;
  compatible = "atkalpha-key";
  pinctrl-names = "default";
  pinctrl-0 = <&pinctrl_key>;
  key-gpio = <&gpio5 1 GPIO_ACTIVE_LOW>;
  status = "okay";
};

下圖所示:

694408cc-d329-11ed-bfe3-dac502259ad0.png

5、編譯設(shè)備樹并把生成的文件復(fù)制到對應(yīng)位置

6972fca4-d329-11ed-bfe3-dac502259ad0.png

6、編寫對應(yīng)的驅(qū)動

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 


/* 添加頭文件 */
#include 
#include 
#include 
#include 


#define CHRDEVBASE_CNT      1    /* 設(shè)備號個數(shù) */
#define CHRDEVBASE_NAME   "chrdevbase"  /* 名字 */


/* 定義按鍵值 */
#define KEY0VALUE 0XF0 /* 按鍵值 */
#define INVAKEY 0X00 /* 無效的按鍵值 */


/* chrdevbase 設(shè)備結(jié)構(gòu)體 */
struct newchr_dev{
  dev_t devid;       /* 設(shè)備號 */
  struct cdev cdev;     /* cdev */
  struct class *class;   /* 類 */
  struct device *device;   /* 設(shè)備 */
  int major;         /* 主設(shè)備號 */
  int minor;         /* 次設(shè)備號 */
  struct device_node *nd; /* 設(shè)備節(jié)點 */
  int key_gpio; /* key 所使用的 GPIO 編號 */
  atomic_t keyvalue; /* 按鍵值 */
};


struct newchr_dev chrdevbase;/* 自定義字符設(shè)備 */


/*
* @description : key 硬件初始化
* @param : 無
* @return : 無
*/
static int key_hal_init(void)
{
  /* 設(shè)置 LED 所使用的 GPIO */
  /* 1、獲取設(shè)備節(jié)點:key */
  chrdevbase.nd = of_find_node_by_path("/key");
  if(chrdevbase.nd == NULL) {
    printk("chrdevbase node cant not found!
");
    return -EINVAL;
  } else {
    printk("chrdevbase node has been found!
");
  }


  /* 2、 獲取設(shè)備樹中的 gpio 屬性,得到 KEY 所使用的 KEY 編號 */
  chrdevbase.key_gpio = of_get_named_gpio(chrdevbase.nd, "key-gpio", 0);
  if(chrdevbase.key_gpio < 0) {
    printk("can't get key-gpio");
    return -EINVAL;
  }
  printk("key-gpio num = %d
", chrdevbase.key_gpio);


  /* 3、初始化 key 所使用的 IO */
  gpio_request(chrdevbase.key_gpio, "key0"); /* 請求 IO */
  gpio_direction_input(chrdevbase.key_gpio); /* 設(shè)置為輸入 */


  return 0;
}


/*
 * @description    : 打開設(shè)備
 * @param - inode   : 傳遞給驅(qū)動的inode
 * @param - filp   : 設(shè)備文件,file結(jié)構(gòu)體有個叫做private_data的成員變量
 *             一般在open的時候?qū)rivate_data指向設(shè)備結(jié)構(gòu)體。
 * @return       : 0 成功;其他 失敗
 */
static int chrdevbase_open(struct inode *inode, struct file *filp)
{
  printk("[BSP]chrdevbase open!
");
  filp->private_data = &chrdevbase; /* 設(shè)置私有數(shù)據(jù) */
  return 0;
}


/*
 * @description    : 從設(shè)備讀取數(shù)據(jù) 
 * @param - filp   : 要打開的設(shè)備文件(文件描述符)
 * @param - buf   : 返回給用戶空間的數(shù)據(jù)緩沖區(qū)
 * @param - cnt   : 要讀取的數(shù)據(jù)長度
 * @param - offt   : 相對于文件首地址的偏移
 * @return       : 讀取的字節(jié)數(shù),如果為負值,表示讀取失敗
 */
static ssize_t chrdevbase_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)
{
  int ret = 0;
  unsigned char value;
  struct newchr_dev *dev = filp->private_data;


  if (gpio_get_value(dev->key_gpio) == 0) { /* key0 按下 */
    while(!gpio_get_value(dev->key_gpio)); /* 等待按鍵釋放 */
    atomic_set(&dev->keyvalue, KEY0VALUE);
  } else { /* 無效的按鍵值 */
    atomic_set(&dev->keyvalue, INVAKEY);
  }


  value = atomic_read(&dev->keyvalue); /* 保存按鍵值 */
  ret = copy_to_user(buf, &value, sizeof(value));
  return ret;


  // printk("chrdevbase read!
");
  // return 0;
}


/*
 * @description    : 向設(shè)備寫數(shù)據(jù) 
 * @param - filp   : 設(shè)備文件,表示打開的文件描述符
 * @param - buf   : 要寫給設(shè)備寫入的數(shù)據(jù)
 * @param - cnt   : 要寫入的數(shù)據(jù)長度
 * @param - offt   : 相對于文件首地址的偏移
 * @return       : 寫入的字節(jié)數(shù),如果為負值,表示寫入失敗
 */
static ssize_t chrdevbase_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)
{
  int retvalue = 0;
  char writebuf[1];


  /* 接收用戶空間傳遞給內(nèi)核的數(shù)據(jù)并且打印出來 */
  retvalue = copy_from_user(writebuf, buf, cnt);
  printk("[BSP]kernel recevdata data:%d!
",writebuf[0]);


  // printk("chrdevbase write!
");
  return 0;
}


/*
 * @description    : 關(guān)閉/釋放設(shè)備
 * @param - filp   : 要關(guān)閉的設(shè)備文件(文件描述符)
 * @return       : 0 成功;其他 失敗
 */
static int chrdevbase_release(struct inode *inode, struct file *filp)
{
  printk("[BSP]release!
");
  return 0;
}


/*
 * 設(shè)備操作函數(shù)結(jié)構(gòu)體
 */
static struct file_operations chrdevbase_fops = {
  .owner = THIS_MODULE,  
  .open = chrdevbase_open,
  .read = chrdevbase_read,
  .write = chrdevbase_write,
  .release = chrdevbase_release,
};


/*
 * @description  : 驅(qū)動入口函數(shù) 
 * @param     : 無
 * @return     : 0 成功;其他 失敗
 */
static int __init chrdevbase_init(void)
{
  /* 初始化硬件 */
  key_hal_init();


  /* 注冊字符設(shè)備驅(qū)動 */
  /* 1、創(chuàng)建設(shè)備號 */
  if (chrdevbase.major) { /* 定義了設(shè)備號 */
    chrdevbase.devid = MKDEV(chrdevbase.major, 0);
    register_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT, CHRDEVBASE_NAME);
  } else { /* 沒有定義設(shè)備號 */
    alloc_chrdev_region(&chrdevbase.devid, 0, CHRDEVBASE_CNT,CHRDEVBASE_NAME); /* 申請設(shè)備號 */
    chrdevbase.major = MAJOR(chrdevbase.devid); /* 獲取主設(shè)備號 */
    chrdevbase.minor = MINOR(chrdevbase.devid); /* 獲取次設(shè)備號 */
  }
  printk("newcheled major=%d,minor=%d
",chrdevbase.major,chrdevbase.minor);


  /* 2、初始化 cdev */
  chrdevbase.cdev.owner = THIS_MODULE;
  cdev_init(&chrdevbase.cdev, &chrdevbase_fops);


  /* 3、添加一個 cdev */
  cdev_add(&chrdevbase.cdev, chrdevbase.devid, CHRDEVBASE_CNT);


  /* 4、創(chuàng)建類 */
  chrdevbase.class = class_create(THIS_MODULE, CHRDEVBASE_NAME);
  if (IS_ERR(chrdevbase.class)) {
    return PTR_ERR(chrdevbase.class);
  }


  /* 5、創(chuàng)建設(shè)備 */
  chrdevbase.device = device_create(chrdevbase.class, NULL,chrdevbase.devid, NULL, CHRDEVBASE_NAME);
  if (IS_ERR(chrdevbase.device)) {
    return PTR_ERR(chrdevbase.device);
  }


  return 0;
}


/*
 * @description  : 驅(qū)動出口函數(shù)
 * @param     : 無
 * @return     : 無
 */
static void __exit chrdevbase_exit(void)
{
  /* 注銷字符設(shè)備 */
  cdev_del(&chrdevbase.cdev);/* 刪除 cdev */
  unregister_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT);/* 注銷設(shè)備號 */


  device_destroy(chrdevbase.class, chrdevbase.devid);/* 銷毀設(shè)備 */
  class_destroy(chrdevbase.class);/* 銷毀類 */


  printk("[BSP]chrdevbase exit!
");
}


/* 
 * 將上面兩個函數(shù)指定為驅(qū)動的入口和出口函數(shù) 
 */
module_init(chrdevbase_init);
module_exit(chrdevbase_exit);


/* 
 * LICENSE和作者信息
 */
MODULE_LICENSE("GPL");
MODULE_AUTHOR("zuozhongkai");

7、編寫應(yīng)用層

#include "stdio.h"
#include "unistd.h"
#include "sys/types.h"
#include "sys/stat.h"
#include "fcntl.h"
#include "stdlib.h"
#include "string.h"


/* 定義按鍵值 */
#define KEY0VALUE 0XF0
#define INVAKEY 0X00


/*
 * @description    : main主程序
 * @param - argc   : argv數(shù)組元素個數(shù)
 * @param - argv   : 具體參數(shù)
 * @return       : 0 成功;其他 失敗
 */
int main(int argc, char *argv[])
{
  int fd, retvalue;
  char *filename;
  unsigned char keyvalue;


  if(argc != 2){
    printf("[APP]Error Usage!
");
    return -1;
  }


  filename = argv[1];


  /* 打開驅(qū)動文件 */
  fd  = open(filename, O_RDWR);
  if(fd < 0){
    printf("[APP]Can't open file %s
", filename);
    return -1;
  }


  /* 循環(huán)讀取按鍵值數(shù)據(jù)!*/
  while(1) {
    read(fd, &keyvalue, sizeof(keyvalue));
    if (keyvalue == KEY0VALUE) { /* KEY0 */
      printf("KEY0 Press, value = %#X
", keyvalue);/* 按下 */
    }
  }


  /* 關(guān)閉設(shè)備 */
  retvalue = close(fd);
  if(retvalue < 0){
    printf("[APP]Can't close file %s
", filename);
    return -1;
  }


  return 0;
}

8、編譯并把生成的文件復(fù)制到對應(yīng)位置

69943536-d329-11ed-bfe3-dac502259ad0.png

9、測試應(yīng)用

69bed494-d329-11ed-bfe3-dac502259ad0.png

測試結(jié)果顯示沒有問題,能正常讀取到按鍵按下時的鍵值,通過在應(yīng)用層中輪詢讀取按鍵按下,當(dāng)按鍵按下時立馬打印出按鍵的鍵值。

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 文件
    +關(guān)注

    關(guān)注

    1

    文章

    551

    瀏覽量

    24559
  • 按鍵
    +關(guān)注

    關(guān)注

    4

    文章

    222

    瀏覽量

    57491
  • Ubuntu
    +關(guān)注

    關(guān)注

    5

    文章

    552

    瀏覽量

    29294
  • GPIO
    +關(guān)注

    關(guān)注

    16

    文章

    1172

    瀏覽量

    51511
  • 設(shè)備樹
    +關(guān)注

    關(guān)注

    0

    文章

    37

    瀏覽量

    3095

原文標題:i.MX6ULL|按鍵輸入捕獲

文章出處:【微信號:玩轉(zhuǎn)單片機,微信公眾號:玩轉(zhuǎn)單片機】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    i.MX6ULL 驅(qū)動開發(fā)7—按鍵輸入捕獲GPIO輸入配置與高低電平讀取

    本篇主要介紹了i.MX6ULL按鍵檢測的使用,主要的知識點是設(shè)備樹的修改,以及GPIO輸入配置與高低電平的讀取。
    的頭像 發(fā)表于 05-24 09:11 ?5937次閱讀
    <b class='flag-5'>i.MX6ULL</b> 驅(qū)動開發(fā)7—<b class='flag-5'>按鍵</b><b class='flag-5'>輸入</b><b class='flag-5'>捕獲</b>與<b class='flag-5'>GPIO</b><b class='flag-5'>輸入</b>配置與高低電平讀取

    使用i.MX6ULL開發(fā)板進行Linux根文件系統(tǒng)的完善

    上一篇推文講了怎么移植根文件系統(tǒng),并在i.MX6ULL開發(fā)板中運行起來,但是會出現(xiàn)一些提示,現(xiàn)在來進行根文件的完善。
    發(fā)表于 10-17 11:13 ?706次閱讀

    移植NXP官方linux 5.4內(nèi)核到i.MX6ULL開發(fā)板

    本文描述移植NXP官方 linux 5.4 內(nèi)核到i.MX6ULL開發(fā)板。
    發(fā)表于 12-19 11:10 ?1896次閱讀

    迅為i.MX6ULL開發(fā)板資料下載,讓Linux學(xué)習(xí)更輕松

    關(guān)機按鍵(ON/OFF)19. 復(fù)位按鍵 引出了一個硬件復(fù)位按鍵(RESET),通過按下此按鍵,可以實現(xiàn) i.MX6ULL 的復(fù)位。 20.
    發(fā)表于 09-23 18:07

    i.MX6ULL開發(fā)板硬件資源

    迅為i.MX6ULL 終結(jié)者開發(fā)板硬件資源非常豐富,幾乎將 i.MX6ULL 芯片的所有資源都擴展引出到底板上了,底板提供了豐富的外設(shè)接口,開發(fā)板的尺寸是 190mm*125mm,充分考慮了人性化設(shè)計,整體顯得十分大。
    發(fā)表于 12-29 06:18

    i.MX6ULL核心板資源

    `STAMP-IMX6ULL-CM是浙江啟揚智能科技有限公司推出的基于 NXP i.MX6ULL 處理器的產(chǎn)品;i.MX 6ULL 是 NXP 推出的 ARM Cortex-A7 系列
    發(fā)表于 07-12 17:50

    初識 i.MX6ULL 寄存器

    i.MX6ULL 電氣屬性寄存器2.4 i.MX6ULL GPIO控制寄存器3. ARM 匯編指令集簡介3.1 處理器工作模式3.2 匯編語言基本格式3.3 偽指令指定“段”3.4 匯編程序入口3.4.1 默認入口
    發(fā)表于 12-20 07:13

    關(guān)于i.MX6ULL配置GPIO

    正如學(xué)習(xí)C語言時寫的第一段代碼都是“HelloWorld!”,接觸一款新的處理器時往往是從點亮一個LED開始;而點亮一個LED,則需要操作這款芯片的GPIO外設(shè)。那么作為廣受歡迎的i.MX6ULL
    發(fā)表于 08-05 10:37

    I.MX6ULL UART傳輸問題求解

    I.MX6ULL UART傳輸問題
    發(fā)表于 04-21 08:09

    珠海明遠智??萍悸?lián)合NXP強勢推出i.MX6ull核心板

    NXP I.MX6ULL是一個高性能、低功耗、高性價比處理器系列,基于ARM Cortex-A7內(nèi)核,主頻可達900MHz。i.MX 6ULL應(yīng)用處理器包括一個集成的電源管理模塊,可以省掉外部
    發(fā)表于 04-24 14:10 ?504次閱讀

    飛凌i.MX6ULL開發(fā)板的評測,再次進階擁有更高的性價比

    處理器MCIMX6Y2開發(fā)設(shè)計,采用先進的ARMCortex-A7內(nèi)核,運行速度高達800MHz。i.MX6ULL應(yīng)用處理器包括一個集成的電源管理模塊,降低了外接電源的復(fù)雜性,并簡化了上電時序。 i.MX6ULL
    發(fā)表于 10-27 11:55 ?1381次閱讀
    飛凌<b class='flag-5'>i.MX6ULL</b>開發(fā)板的評測,再次進階擁有更高的性價比

    基于NXP i.MX6ULL處理器的FETMX6ULL-C核心板

    “性價比高,功能接口豐富,資料齊全,穩(wěn)定性強”這是許多用戶對飛凌FETMX6ULL-S核心板的評價。作為NXP公司一顆經(jīng)典的MPU,i.MX6ULL的市場認可度無需多言。而作為NXP公司的金牌
    發(fā)表于 04-11 15:05 ?1057次閱讀
    基于NXP <b class='flag-5'>i.MX6ULL</b>處理器的FETMX<b class='flag-5'>6ULL</b>-C核心板

    基于i.MX6ULL點亮LED

    都說入門一款芯片的第一步是點亮LED,但是i.MX6ULL入門門檻比較高,特別是通過自學(xué)入門的,這個系列已經(jīng)寫了好久了,最近打算在項目不急的時候加快一下學(xué)習(xí)進度,現(xiàn)在就開始學(xué)習(xí)一下怎么點亮一個LED
    的頭像 發(fā)表于 03-06 09:09 ?689次閱讀

    【北京迅為】i.MX6ULL開發(fā)板移植 Debian 文件系統(tǒng)

    【北京迅為】i.MX6ULL開發(fā)板移植 Debian 文件系統(tǒng)
    的頭像 發(fā)表于 02-10 15:34 ?1011次閱讀
    【北京迅為】<b class='flag-5'>i.MX6ULL</b>開發(fā)板移植 Debian 文件系統(tǒng)

    基于i.MX6ULL的掉電檢測設(shè)計與軟件測試

    基于i.MX6ULL的掉電檢測設(shè)計與軟件測試基于i.MX6ULL平臺設(shè)計實現(xiàn)掉電檢測功能,首先選擇一路IO,利用IO電平變化觸發(fā)中斷,在編寫驅(qū)動時捕獲該路GPIO的中斷,然后在中斷響應(yīng)
    的頭像 發(fā)表于 11-09 10:40 ?683次閱讀
    基于<b class='flag-5'>i.MX6ULL</b>的掉電檢測設(shè)計與軟件測試