树莓派linux驱动学习之LED控制
2013-11-21 14:54
567 查看
前面我们编写了hello world的程序,接下来继续研究GPIO功能,通过GPIO来控制LED的亮灭,这在单片机中应该算是十分简单的一个程序了,但是在Linux系统中控制GPIO没有那么简单,难点就在于GPIO地址的获取,也是我一直在纠结的问题。
但是在芯片资料的最开始,有提到芯片内部已经把上图中的物理总线地址抽象到了面对操作系统的物理地址:
所以,我们在编写驱动程序的时候,IO空间的起始地址是0x20000000,加上GPIO的偏移量2000000,所以GPIO的物理地址应该是从0x20200000开始的,然后在这个基础上进行Linux系统的MMU内存虚拟化管理,银蛇到虚拟地址上。
驱动代码:
应用测试程序:
分别编译,插入模块,然后运行测试程序,可以控制LED的亮灭了。
----------------------------------------------------------------
欢迎大家转载我的文章。
转载请注明:转自古-月
http://blog.csdn.net/hcx25909
欢迎继续关注我的博客
一、GPIO地址
我看了中嵌的嵌入式开发视频,里面使用三星2440控制LED的亮灭,但是驱动程序中没有写清楚具体的底层是如何实现的,这也是我查找的重点。我首先翻阅了树莓派CPU(bcm2835)的芯片手册,查到了GPIO的物理地址:但是在芯片资料的最开始,有提到芯片内部已经把上图中的物理总线地址抽象到了面对操作系统的物理地址:
所以,我们在编写驱动程序的时候,IO空间的起始地址是0x20000000,加上GPIO的偏移量2000000,所以GPIO的物理地址应该是从0x20200000开始的,然后在这个基础上进行Linux系统的MMU内存虚拟化管理,银蛇到虚拟地址上。
二、硬件平台
我在树莓派的扩展口的GPIO 17上接上了一个LED:三、编写驱动代码
一般的设备驱动我们需要设置主设备号和次设备号,在编写应用程序的时候还要生成设备文件,比较麻烦。Linux针对像LED这样的操作,有一种设备叫做混杂设备:是一种特殊的字符设备,杂设备早已经存在,是为了给开发者一个较为简单的操作方式,因为不用再重新申请一个设备号了(misc就是混杂设备的意思)。驱动代码:
#include <linux/miscdevice.h> #include <linux/delay.h> #include <asm/irq.h> #include <linux/kernel.h> #include <linux/module.h> #include <linux/init.h> #include <linux/mm.h> #include <linux/fs.h> #include <linux/types.h> #include <linux/delay.h> #include <linux/moduleparam.h> #include <linux/slab.h> #include <linux/errno.h> #include <linux/ioctl.h> #include <linux/cdev.h> #include <linux/string.h> #include <linux/list.h> #include <linux/pci.h> #include <asm/uaccess.h> #include <asm/atomic.h> #include <asm/unistd.h> #include <asm/io.h> #include <asm/uaccess.h> #include <linux/ioport.h> #include "bcm2835.h" // Blinks on RPi Plug P1 pin 11 (which is GPIO pin 17) #define PIN RPI_GPIO_P1_11 int open_state = 0; //文件打开状态 static int leds_open(struct inode *inode, struct file *filp) { if(open_state == 0) { open_state = 1; printk("Open file suc!\n"); return 0; } else { printk("The file has opened!\n"); return -1; } } static int leds_ioctl(struct file*filp, unsigned int cmd, unsigned long arg) { switch(cmd) { case 0: bcm2835_gpio_clr(PIN); printk("LED OFF!\n"); break; case 1: bcm2835_gpio_set(PIN); printk("LED ON!\n"); break; default: return-EINVAL; } return 0; } static int leds_release(struct inode *inode, struct file *filp) { if(open_state == 1) { open_state = 0; printk("close file suc!\n"); return 0; } else { printk("The file has closed!\n"); return -1; } } static const struct file_operations leds_fops = { .owner = THIS_MODULE, .open = leds_open, .unlocked_ioctl = leds_ioctl, .release = leds_release, }; static struct miscdevice misc = { .minor =MISC_DYNAMIC_MINOR, .name ="my_leds", .fops =&leds_fops, }; static int __init leds_init(void) { int ret; //注册混杂设备 ret =misc_register(&misc); //配置功能选择寄存器为输出 bcm2835_gpio_fsel(PIN, BCM2835_GPIO_FSEL_OUTP); //设置输出电平为高电平,LED亮 bcm2835_gpio_set(PIN); printk("ledsinit.\n"); return ret; } static void leds_exit(void) { //LED灭 bcm2835_gpio_clr(PIN); misc_deregister(&misc); printk("leds_exit\n"); } module_init(leds_init); module_exit(leds_exit); MODULE_AUTHOR("Hu Chunxu"); MODULE_LICENSE("GPL");硬件相关操作:
#include <linux/miscdevice.h> #include <linux/delay.h> #include <asm/irq.h> #include <linux/kernel.h> #include <linux/module.h> #include <linux/init.h> #include <linux/mm.h> #include <linux/fs.h> #include <linux/types.h> #include <linux/delay.h> #include <linux/moduleparam.h> #include <linux/slab.h> #include <linux/errno.h> #include <linux/ioctl.h> #include <linux/cdev.h> #include <linux/string.h> #include <linux/list.h> #include <linux/pci.h> #include <asm/uaccess.h> #include <asm/atomic.h> #include <asm/unistd.h> #include <asm/io.h> #include <asm/uaccess.h> #include <linux/ioport.h> #include "bcm2835.h" int bcm2835_gpio_fsel(uint8_t pin, uint8_t mode) { //初始化GPIOB功能选择寄存器的物理地址 volatile uint32_t * bcm2835_gpio = (volatile uint32_t *)ioremap(BCM2835_GPIO_BASE, 16); volatile uint32_t * bcm2835_gpio_fsel = bcm2835_gpio + BCM2835_GPFSEL0/4 + (pin/10); uint8_t shift = (pin % 10) * 3; uint32_t value = mode << shift; *bcm2835_gpio_fsel = *bcm2835_gpio_fsel | value; printk("fsel address: 0x%lx : %x\n", bcm2835_gpio_fsel, *bcm2835_gpio_fsel); return 0; } int bcm2835_gpio_set(uint8_t pin) { //GPIO输出功能物理地址 volatile uint32_t * bcm2835_gpio = (volatile uint32_t *)ioremap(BCM2835_GPIO_BASE, 16); volatile uint32_t * bcm2835_gpio_set = bcm2835_gpio + BCM2835_GPSET0/4 + pin/32; uint8_t shift = pin % 32; uint32_t value = 1 << shift; *bcm2835_gpio_set = *bcm2835_gpio_set | value; printk("set address: 0x%lx : %x\n", bcm2835_gpio_set, *bcm2835_gpio_set); return 0; } int bcm2835_gpio_clr(uint8_t pin) { //GPIO清除功能物理地址 volatile uint32_t * bcm2835_gpio = (volatile uint32_t *)ioremap(BCM2835_GPIO_BASE, 16); volatile uint32_t * bcm2835_gpio_clr = bcm2835_gpio + BCM2835_GPCLR0/4 + pin/32; uint8_t shift = pin % 32; uint32_t value = 1 << shift; *bcm2835_gpio_clr = *bcm2835_gpio_clr | value; printk("clr address: 0x%lx : %x\n", bcm2835_gpio_clr, *bcm2835_gpio_clr); return 0; }
应用测试程序:
#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/ioctl.h> int main(int argc, char **argv) { int on; int fd; if (argc != 2 || sscanf(argv[1],"%d", &on) != 1 ||on < 0 || on > 1 ) { fprintf(stderr, "Usage:%s 0|1\n",argv[0]); exit(1); } fd = open("/dev/my_leds", 0); if (fd < 0) { perror("open device leds"); exit(1); } /*通过ioctl来控制灯的亮、灭*/ if(on){ printf("turn on leds!\n"); ioctl(fd, 1); } else { printf("turn off leds!\n"); ioctl(fd, 0); } close(fd); return 0; }
分别编译,插入模块,然后运行测试程序,可以控制LED的亮灭了。
----------------------------------------------------------------
欢迎大家转载我的文章。
转载请注明:转自古-月
http://blog.csdn.net/hcx25909
欢迎继续关注我的博客
相关文章推荐
- 树莓派linux驱动学习之LED控制
- 树莓派linux驱动学习之LED控制
- 树莓派linux驱动学习之LED控制
- linux字符驱动学习实践1(简单控制LED灯)
- Linux button按键驱动 多次中断控制相应LED灯亮灭闪
- 嵌入式Linux驱动学习之路(十)字符设备驱动-my_led
- 嵌入式Linux应用学习(一)------QT控制LED设备硬件
- Linux学习:LED字符设备驱动
- linux驱动学习之并发和竞争控制
- Android下led控制(下)--Linux驱动部分--script与gpio(全志)
- 嵌入式Linux学习入门:控制LED灯
- linux驱动学习之LED篇(一)
- 嵌入式系统学习——S3C2451之linux驱动led
- Linux字符驱动学习之LED
- 树莓派学习笔记——Linux I2C驱动说明
- 树莓派学习笔记——Linux I2C驱动说明
- LED PWM控制芯片PCA9685的Linux 驱动
- S3c6410 linux内核移植(10)---添加充电控制脚驱动(LED驱动)
- linux驱动学习之并发和竞争控制
- 一步一步学习 Linux 驱动之设备控制接口函数(ioctl 函数)