目前我所知道的在linux下操作GPIO有两种方法:
程序是转来的,感谢作者的辛苦整理。博文地址:www.cnblogs.com/lidabo/p/5319219.html,这两种方法都是通过直接写物理地址的方法控制GPIO的状态。第一种是在驱动代码里做好映射、配置,然后应用层通过IOCTL控制GPIO的读写,第二种是直接就mmap了,直接写地址了。
1. 编写驱动,这当然要熟悉linux下驱动的编写方法和技巧,在驱动里可以使用ioremap函数获得GPIO物理基地址指针,然后使用这个指针根据ioctl命令进行GPIO寄存器的读写,并把结果回送到应用层。这里提供一点程序片断供大家参考:
- int init_module(void){
- printk(KERN_ALERT "ioctl load.\r\n");
- register_chrdev(254,"ioreg",&fops);
- stbgpio = (STBX25XX_GPIO_REG *)ioremap(GPIO_BASE,GPIO_LEN);
- if(stb_gpio == NULL){
- printk(KERN_ALERT "can''t get io base.\r\n");
- return -1;
- }
- return 0;
- }
- int io_ioctl (struct inode *inode, struct file *filp,unsigned int cmd, unsigned long arg){
- unsigned long uGpio;
- printk(KERN_ALERT "io_ioctl cmd=%04x,arg=%04x.\r\n",cmd,(int)arg);
- switch(cmd){
- case SET_IO_DIR:{
- printk(KERN_ALERT "SET_IO_DIR\r\n");
- break;
- }
- case SET_IO_VALUE:{
- printk(KERN_ALERT "SET_IO_VALUE\r\n");
- break;
- }
- case GET_IO_VALUE:{
- printk(KERN_ALERT "GET_IO_VALUE\r\n");
- uGpio = stb_gpio->GPI;
- printk(KERN_ALERT "GPIO = %08x",(int)uGpio);
- copy_to_user((void *)arg,(const void *) &uGpio,sizeof(uGpio));
- break;
- }
- case GET_IO_DIR:{
- printk(KERN_ALERT "GET_IO_DIR\r\n");
- break;
- }
- }
- return 0;
- }
2. 在应用层使用mmap函数在应用层获得GPIO物理基地址对应的虚拟地址指针,然后使用这个指针来读写GPIO寄存器,这里提供一点程序片断供大家参考:
- char dev_name[] = "/dev/mem";
- GPIO_REGISTER *gpio_base;
- fd = open(dev_name,O_RDWR);
- if(fd<0){
- printf("open %s is error\n",dev_name);
- return -1 ;
- }
- gpio_base = (GPIO_REGISTER *)mmap( 0, 0x32, PROT_READ | PROT_WRITE, MAP_SHARED,fd, 0x40060000 );
- if(gpio_base == NULL){
- printf("gpio base mmap is error\n");
- close(fd);
- return -1;
- }
- gpio_base->or = (gpio_base->or & 0x7fffffff);