1. 确定硬件连接

  主要电路连接如下:

  从电路图中可以发现开发板上6个按键连接到了GPN0~5,当按键弹起时IO状态应为高电平,当按键按下时IO口状态为低电平。

2. 确定寄存器

  下图为寄存器的名称和地址:

  

  下图为配置寄存器的描述:

  

  下图为数据寄存器和上/下拉寄存器:

  

  需要注意的是,GPN默认是下拉使能,从电路图中判断按键应该配置为上拉使能,其他没什么好说的。

3. 编写代码

  key.c

 1 #include <linux/init.h>
 2 #include <linux/module.h>
 3 #include <linux/io.h>
 4 #include <linux/fs.h>
 5 #include <linux/miscdevice.h>
 6 #include <asm/uaccess.h>
 7 
 8 #define GPNCON    0x7F008830
 9 #define GPNDAT    0x7F008834
10 #define GPNPUD    0x7F008838
11 
12 static int *__gp_gpndat = NULL;
13 
14 static int key_open (struct inode *p_node, struct file *p_file)
15 {
16     int *p_gpncon = NULL;
17     int *p_gpnpud = NULL;
18     int temp;
19 
20     p_gpncon = ioremap(GPNCON, 4);
21     __gp_gpndat = ioremap(GPNDAT, 4);
22     p_gpnpud = ioremap(GPNPUD, 4);
23 
24     temp = readl(p_gpncon);
25     temp &= ~(0xFFF);
26     writel(temp, p_gpncon);
27 
28     temp = readl(p_gpnpud);
29     temp &= ~(0xFFF);
30     temp |= 0xAAA;
31     writel(temp, p_gpnpud);
32 
33     return 0;
34 }
35 
36 static ssize_t key_read (struct file *p_file, char *p_dat, size_t size, loff_t *p_loff)
37 {
38     int key;
39     char i;
40 
41     key = readl(__gp_gpndat);
42     if ((key & 0x3F) != 0x3F) {
43         for (i = 1; i <= 6; i++) {
44             if ((key & 1) == 0) {
45                 if (copy_to_user(p_dat, &i, 1)) {
46                     return -1;
47                 }
48                 return 1;
49             }
50             key >>= 1;
51         }
52     }
53 
54     return 0;
55 }
56 
57 static struct file_operations fops = {
58     .owner = THIS_MODULE,
59     .open = key_open,
60     .read = key_read,
61 };
62 
63 static struct miscdevice key_dev = {
64     .minor = MISC_DYNAMIC_MINOR,
65     .name = "mykey",
66     .fops = &fops,
67 };
68 
69 static int __init key_init (void)
70 {
71     misc_register(&key_dev);
72 
73     return 0;
74 }
75 
76 static void __exit key_exit (void)
77 {
78     misc_deregister(&key_dev);
79 }
80 
81 module_init(key_init);
82 module_exit(key_exit);
83 
84 MODULE_LICENSE("GPL");

  key_app.c

 1 #include <stdio.h>
 2 #include <sys/types.h>
 3 #include <sys/stat.h>
 4 #include <fcntl.h>
 5 
 6 int main(int argc, const char *argv[])
 7 {
 8     int fd;
 9     char key_val;
10     int key_push_flag = 0;
11     int key_val_temp;
12     int cnt;
13 
14     fd = open("/dev/mykey", O_RDWR);
15 
16     while (1) {
17         cnt = read(fd, &key_val, 1);
18         if ((key_push_flag == 0) && (1 == cnt)) {
19             key_push_flag = 1;
20 
21             key_val_temp = key_val;
22             printf("key %d down\n", key_val_temp);
23         } else if (cnt == 0) {
24             key_push_flag = 0;
25         }
26     }
27 
28     return 0;
29 }

Makefile

 1 obj-m := key.o
 2 KDIR = /home/linux/zkf/6410/urbetter-linux2.6.28-v1.0    
 3 
 4 all:
 5     make -C $(KDIR) M=$(PWD) modules CROSS_COMPILE=arm-none-linux-gnueabi- ARCH=arm
 6     arm-none-linux-gnueabi-gcc key_app.c -o key_app
 7     sudo cp key_app ~/zkf/6410/rootfs/home/test
 8     sudo cp key.ko ~/zkf/6410/rootfs/home/test
 9 clean:
10     rm *.o *.mod.c *.symvers *.order

4. 运行

  加载模块:insmod key.ko

  运行key_app,现象如下图: