Am335x 下GPIO控制实例-驱动程序(转)

看了这么多的资料,现在决定上手了,下面将用两种方式来实现对GPIO 117的控制
1,用直接添加到内核的方式,实现MISC的驱动(misc_register)
2,用手工安装的方式,实现简单字符设备驱动(register_chrdev)

实现前提:当前所用的GPIO没有被其它设备所使用,大家可以用我前面BLOG说的方式查看GPIO的使用情况,当前我所用的GPIO本来是bluetooth的开关,需要屏蔽一个函数。不然后面的驱动申请IO都会失败。
函数为Board-am335xevm.c 中的wl12xx_bluetooth_enable();

一,MISC驱动的实现
1,参考linux SDK for AM335x Ver 1.1.pdf P28,添加kernel 配置选项
  打开/driver/input/misc/Kconfig并添加:

[objc] view plain copy

  1. config INPUT_GPIOTEST
  2. bool "Gpio 117 test"
  3. help
  4. Just test the Gpio 117 status

打开/driver/input/misc/Makefile并添加:

[plain] view plain copy

  1. obj-$(CONFIG_INPUT_GPIOTEST)+=GpioTestDriver.o

2,实现GpioTestDriver.c

[objc] view plain copy

  1. #include <linux/gpio.h>
  2. #include <linux/module.h>
  3. #include <linux/kernel.h>
  4. #include <linux/moduleparam.h>
  5. #include <linux/delay.h>
  6. #include <linux/types.h>
  7. #include <linux/miscdevice.h>
  8. #include <linux/device.h>
  9. #include <linux/fs.h>
  10. #include <linux/init.h>
  11. #define TEST_IO_NUM(117)
  12. #define NAME_MISC"GpioTest"
  13. #define NAME_MOUDULE"GpioTest1"
  14. #define USE_MISC_MODE1
  15. static int major = 251;
  16. void GpioTest(void);
  17. static long GpioIOctl(struct file *filp, unsigned cmd, unsigned long arg)
  18. {
  19. GpioTest();
  20. return 1;
  21. }
  22. void GpioTest(void)
  23. {
  24. int iCount = 0;
  25. for(iCount = 0; iCount <=20; iCount++ )
  26. {
  27. if(iCount%2 == 0)
  28. {
  29. gpio_direction_output(TEST_IO_NUM, 1);
  30. printk(KERN_INFO"#######IO117 statu is high.\r\n");
  31. }
  32. else
  33. {
  34. gpio_direction_output(TEST_IO_NUM, 0);
  35. printk(KERN_INFO"#######IO117 statu is low.\r\n");
  36. }
  37. mdelay(3000);
  38. }
  39. printk(KERN_INFO"#######App run over!");
  40. }
  41. static int GpioOpen(struct inode *inode, struct file *file)
  42. {
  43. int iRen = -1;
  44. iRen = gpio_request(TEST_IO_NUM, "IO117");
  45. if(iRen < 0)
  46. {
  47. printk(KERN_INFO"#######Failed to request the IO117!");
  48. }else
  49. {
  50. printk(KERN_INFO"#######Success to request the IO117");
  51. }
  52. return iRen;
  53. }
  54. static int GpioClose(struct inode *inode, struct file *file)
  55. {
  56. printk(KERN_INFO"#######Free the IO117");
  57. gpio_free(TEST_IO_NUM);
  58. return 1;
  59. }
  60. //****entry point for TEST GPIO module
  61. static const struct file_operations gpio_test_driver = {
  62. .owner = THIS_MODULE,
  63. .unlocked_ioctl= GpioIOctl,
  64. .llseek = no_llseek,
  65. .open = GpioOpen,
  66. .release = GpioClose,
  67. };
  68. #if USE_MISC_MODE
  69. static struct miscdevice gpiotest_misc_device = {
  70. .minor    = MISC_DYNAMIC_MINOR,
  71. .name     = NAME_MISC,
  72. .fops     = &gpio_test_driver,
  73. };
  74. #endif
  75. static int __init GpioTestInit(void)
  76. {
  77. int iRet;
  78. printk(KERN_INFO"#######GpioTest modules is install!\r\n");
  79. #if USE_MISC_MODE
  80. iRet = misc_register(&gpiotest_misc_device);
  81. if (iRet) {
  82. printk(KERN_INFO"#######unable to register a misc device\r\n");
  83. return iRet;
  84. }
  85. #else
  86. iRet = register_chrdev(major, NAME_MOUDULE, &gpio_test_driver);
  87. if (iRet < 0) {
  88. printk(KERN_INFO"#######unable to register a chr device\r\n");
  89. return iRet;
  90. }
  91. #endif
  92. return iRet;
  93. }
  94. static void __exit GpioTestExit(void)
  95. {
  96. #if USE_MISC_MODE
  97. misc_deregister(&gpiotest_misc_device);
  98. #else
  99. unregister_chrdev(major, NAME_MOUDULE);
  100. #endif
  101. printk(KERN_INFO"#######GpioTest modules is exit!\r\n");
  102. }
  103. module_init(GpioTestInit);
  104. module_exit(GpioTestExit);
  105. MODULE_AUTHOR("david.hu<[email protected]>");
  106. MODULE_LICENSE("GPL");
  107. MODULE_DESCRIPTION("Gpio117 Test driver");

3,直接编译:
make uImage
拷到小板上升级运行
注意启动的过程有打印:
[    3.730712] #######GpioTest modules is install!
这里表示我们的驱动已经合入NK里去了,当然我们也可以命令:ls /dev,可以看到有GpioTest这个存在
4,写测试APP

[objc] view plain copy

  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <sys/ioctl.h>
  4. #include <unistd.h>
  5. #include <sys/stat.h>
  6. #include <linux/input.h>
  7. #include <fcntl.h>
  8. int main(int argc, charchar *argv)
  9. {
  10. int fd;
  11. fd = open("/dev/GpioTest", O_RDWR);
  12. if(fd < 0)
  13. {
  14. printf("***Can‘t open the gpiotest!\r\n");
  15. return -1;
  16. }
  17. ioctl(fd, 0, 0);
  18. close(fd);
  19. printf("***App run over!\r\n");
  20. return 1;
  21. }

将编译的.out文件拷到小机上面运行,看是不是会打印正确的结果。

二,字符设备驱动的实现
1,代码的实现,请将上面MISC的代码里#define USE_MISC_MODE1改成0
2,makefile的实现

[plain] view plain copy

  1. KERNEL_DIR := /home/ding/workdir/david/EVMBoard/board-support/linux-3.2
  2. PLATFORM := "am335x-evm"
  3. MACHINE_NAME := "am335x"
  4. # If CROSS_COMPILE is not set by Rules.make then set a sane default
  5. CROSS_COMPILE ?= arm-arago-linux-gnueabi-
  6. export CROSS_COMPILE
  7. obj-m := GpioTestDriver.o
  8. MAKE_ENV = ARCH=arm
  9. PWD := $(shell pwd)
  10. all:
  11. $(MAKE) EXTRA_CFLAGS="$(EXTRA_CFLAGS)" -C $(KERNEL_DIR) $(MAKE_ENV) \
  12. M=$(PWD) modules
  13. clean:
  14. rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions *.symvers

3,将编译的ko拷入小机,然后命令:

[plain] view plain copy

  1. insmod ./GpioTestDriver.ko
  2. lsmod
  3. mknod /dev/GpioTestDriver c 251 0

4,实现APP的代码
将上面MISC的代码作如下修改:

[plain] view plain copy

  1. fd = open("/dev/GpioTestDriver", O_RDWR);

5,运行APP查看结果

总结:两个驱动代码实现差不多,但是步骤不一样,主要体现在模块需要安装。MISC会自动创建设备文件,它的主设备号是10,字符设备需要我们来指定。
可安装的字符设备对驱动的编写测试是非常有帮助的。

http://blog.csdn.net/hudaweikevin/article/details/16826995

时间: 2024-10-14 17:40:59

Am335x 下GPIO控制实例-驱动程序(转)的相关文章

python环境下实现OrangePi Zero寄存器访问及GPIO控制

最近入手OrangePi Zero一块,程序上需要使用板子上自带的LED灯,在网上一查,不得不说OPi的支持跟树莓派无法相比.自己摸索了一下,实现简单的GPIO控制方法,作者的Zero安装的是Armbian系统,使用python写了一个读写寄存器的简单模块,通过这个模块,即可实现对GPIO的控制. 作者以前使用过STM32的MCU,这类MCU,如果要实现对GPIO的控制,只需要根据datasheet查找相应GPIO寄存器并进行配置,即可实现IO控制,例如,要将内存地址为0x12345678的寄存

Linux下GPIO驱动

编写驱动程序,首先要了解是什么类型的设备.linux下的设备分为三类,分别为:字符设备,块设备和网络设备.字符设备类型是根据是否以字符流为数据的交换方式,大部分设备都是字符设备,如键盘,串口等,块设备则是以块为单位进行管理的设备,如,磁盘.网络设备就是网卡等. 其次要了解应用程序和驱动程序的区别,两者的主要区别分为以下三点: 1入口函数的任务不相同,应用程序完成一个任务,驱动只完成初始化工作,比如中断 申请,寄存器设置,定时器设置. 2运行时的cpu模式不相同,驱动具有很高的权限,应用程序是在用

BeagleBone black的GPIO控制初探

刚刚读大学的时候就开始接触了 51,后来是stm32, 感觉MCU都是很好玩的 . 做过一些东西,循迹小车之类的,但是我的外围电路总是出问题,所以有很长一段时间没碰这东西了,现在呢到手一块bbbalck  熟悉了一个月  感觉用linux还是很不舒服的(没我pc跑得ubuntu舒服 很多东西看了半天设置不出来  ),于是从ti下载了starterware  结果发现没有bbblack 得从github上clone git clone https://github.com/embest-tech/

(12)树莓派B+ GPIO控制四驱车

吐槽一下:本人还是个小白,CSDN文章分类不知道选哪个 如果你手上的材料是五线四相步进电机驱动板ULN2003+五线四相步进电机28BYJ-48-5V,那么应该算是非常幸运,因为这两样是配套的,一个驱动板可以接一个步进电机,实现正转.反转.加速.减速的功能(个人猜测).很多人因为在购置树莓派外围设备时,没有考虑清楚,随随便便就买了电机和驱动板,导致无法正常驱动电机,譬如我买的是带4个直流电机的四驱车,其中的电机只有两条线接正极和负极,以及ULN2003五线四相步进电机驱动板,这两样是不配套的,不

树莓派GPIO控制

陈拓[email protected] 2018.06.09/2018.06.10 0.  概述 本文介绍树莓派 Zero W的GPIO控制,并用LED看效果. 0.1 树莓派GPIO编号方式 功能物理引脚 从左到右,从上到下:左边基数,右边偶数:1-40 BCM 编号侧重CPU寄存器,根据BCM2835的GPIO寄存器编号. wiringPi 编号侧重实现逻辑,把扩展GPIO端口从0开始编号,这种编号方便编程.如图 WiringPi一栏. 操作GPIO时一定先要清楚使用那一套编号. 1.  准

Java常用的设计模式09:单例模式的强化(控制实例个数n)

1. 单例模式的本质: 控制实例数目(目的节约资源) 2. 单例模式体现的一些思想: (1)延迟装载(Lazy Load):懒汉式 (2)缓存:饿汉式 3. 单例模式的变形使用: 控制使用实例个数为3个(实例池): 1 public class Person { 2 //定义用来缓存数据的容器 3 private static Map map = new HashMap(); 4 //用来记录当前正在使用哪一个实例,初始化值为1 5 private static int num = 1; 6 /

编辑模式下,控制对象移动

有时候我们可能会有这样的需求,就是在编辑模式下,控制移动场景中的物体,这里面有两个点要解决: (1)怎么在编辑模式下运行一个脚本: (2)怎么有效地响应鼠标按键. 第一个问题可以使用编辑器属性ExecuteInEditMode让脚本在编辑模式也能实时地更新.下面的代码用来控制物体的移动,并且可以在编辑模式下运行: using UnityEngine; using System.Collections; #if UNITY_EDITOR using UnityEditor; #endif [Exe

一个多线程下的单实例实现故障

最近在asp.net项目中碰到的一个问题. 一个编码器模块使用单实例实现,结果碰到了编码ID重复的异常. public class IdGenerator { private static IdGenerator single; public static IdGenerator Single { get { if (null == single) { single = new IdGenerator(); } return single; } } protected IdGenerator()

纯css+js下拉菜单实例代码

纯css+js下拉菜单实例代码 分享一个css+js下拉菜单代码,js+css实现的简单下拉菜单,兼容性不错. 例子:<html xmlns="http://www.w3.org/1999/xhtml" lang="zh-CN"> <head> <meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> <