zoukankan      html  css  js  c++  java
  • 【玩转开源】BananaPi R2——移植RPi.GPIO 到 R2

    1. 首先给大家介绍一下什么是RPi.GPIO.

        简单去讲,RPi.GPIO就是一个运行在树莓派开发板上可以通过Python去控制GPIO的一个中间件。

        现在我这边做了一个基础功能的移植,接下来大家可以跟着我去学习一下RPi.GPIO是如何通过Python去实现控制开发板上的GPIO的。

    2. 看一下效果图:

    2.1 硬件实物运行效果

        

    2.2 执行Python脚本打印的log

        

    3. 那么RPi.GPIO在R2上是如何使用的呢?

    3.1 首先在R2上面运行一个Ubuntu镜像,然后下载代码:git clone https://github.com/BPI-SINOVOIP/RPi.GPIO

    这里如果不清楚如何安装一个Ubuntu镜像到R2,欢迎留言。

    3.2 安装Python环境

    1 sudo apt-get update
    2 sudo apt-get install python-dev python3-dev

    3.3 安装中间件

    sudo python setup.py install 
    或者是 sudo python3 setup.py install 来安装
    (前者是使用Python2.X,后者是Python3.X)

        安装参考:http://wiki.banana-pi.org/Getting_Started_with_R2#Install_RPi.GPIO(偷偷告诉你们这个wiki我也在维护哦,欢迎大家留言提建议)

     3.4 安装完成后进入到R2 Ubuntu下的路径:cd /usr/local/bin,你会发现有个g40.py的python文件,我们可以打开简单看一下(这里代码的注释,是我加的,目的是为了让大家更好理解这段Python程序):

    #!/usr/bin/python
    import RPi.GPIO as GPIO  #这里导入RPi.GPIO模块,重命名为GPIO
    import time              #为了方便理解import xxx, 大家可以当作是C语言的#include<xxx.h>
    
    #定义两组要控制的Led GPIO pin数组 phy_led2
    = [8, 10, 12, 16, 18, 22, 24, 26, 28, 32, 36, 38, 40, 37, 35, 33, 31, 29, 27, 23, 21, 19, 15, 13, 11, 7, 5, 3]; phy_led = [8, 10, 12, 16, 18, 22, 24, 26, 32, 36, 38, 40, 37, 35, 33, 31, 29, 23, 21, 19, 15, 13, 11, 7, 5, 3]; print 'Pi Board Information' print '---------------------'
    #这里是一个for循环, 从GPIO.RPI_INFO.items里面读取数据,并打印出来
    for key,val in GPIO.RPI_INFO.items(): print '%s => %s'%(key,val)
    #读取键盘输入 response
    = raw_input(' Is this board info correct (y/n) ? ').upper()
    #调用GPIO.setmode方法,并传入参数GPIO.BOARD
    #(这里大家肯定看得一愣一愣的,这些GPIO.XXX方法,参数究竟是在哪里定义的,不要急,后面就会介绍) GPIO.setmode(GPIO.BOARD)
    #读取phy_led数组,并调用GPIO.setup方法,传入参数pin和GPIO.OUT
    for pin in phy_led: print pin, "GPIO.setup GPIO.OUT" GPIO.setup(pin, GPIO.OUT)
    #这里是一个死循环,意思就是设置GPIO循环输出高低电平不断打开,关闭LED
    while True: for pin in phy_led: GPIO.output(pin, True) print 'on ', pin time.sleep(.1) for pin in phy_led: GPIO.output(pin, False) print 'off ',pin time.sleep(.1) #如果大家对Python不太熟悉,建议可以先去学习一些基础的语法,这样有助于理解。
    #不过不熟悉也没有关系,这里我也会尽量讲明白这个Python文件是在做什么。

    3.5 最上层的接口看完了,接下来我们该看中间件是如何实现上面这些接口的调用的

        下载代码:git clone https://github.com/BPI-SINOVOIP/RPi.GPIO

        代码下载完成后,我们打开先简单看一看,我们从g40.py里面调用的第一个接口看起:GPIO.RPI_INFO.items()

    我的开发环境是Ubuntu,使用的IDE是SourceInsight

        我们先搜索一下接口RPI_INFO:

       

        发现这个接口三在Py_gpio.c里面有出现,我们点进去看一下:

         

        原来这个接口在这里,那这里的PyModule_AddObject是干什么的呢?先看一下官方解释:

        这个接口实际上是Python调用外部代码的一种方式,简单理解就是可以通过 “Module.name” 的方式来调用 "value".

        然后我们再来看一下g40.py里面的这段代码:

    for key,val in GPIO.RPI_INFO.items():
        print '%s => %s'%(key,val)

        Module的名称为GPIO,name是"RPI_INFO",这里使用GPIO.RPI_INFO实际上就调用到了value,根据上面代码的定义,这个value是board_info,那么这段代码实际上就是调用board_info,接下来我们再看看board_info是怎么定义的.

    board_info = Py_BuildValue("{sissssssssss}",
                                  "P1_REVISION",rpiinfo.p1_revision,
                                  "REVISION",&rpiinfo.revision,
                                  "TYPE",rpiinfo.type,
                                  "MANUFACTURER",rpiinfo.manufacturer,
                                  "PROCESSOR",rpiinfo.processor,
                                  "RAM",rpiinfo.ram);

        这里可以看到board_info是从Py_BuildValue这里获取的值,那么Py_BuildValue又是做什么的呢?先看一下官方的解释:

         意思就是获取C语言格式的字符串,看起来这里Python和C开始有联系咯,我们看看最开始的打印:

        

        看到这里的打印,和上面Py_BuildValue()里面的参数对比一下,是不是开始有感觉了;没错这里已经成功从Python调用到C了.

        我们接下来再看看g40.py,代码马上执行到:GPIO.setmode(GPIO.BOARD),对应的我们去看源码是怎么样的:

        首先我们找一下这个BOARD参数,可以看到这个参数是一个int类型的变量;

      

        接下来我们再看看setmode方法:

        这里可以看到setmode方法实际上是调用的py_setmode,我们再看看这个py_setmode是怎么定义的:

        看到这里实际上我们会发现,其实就是C的写法了,简单去看就是返回一个PyObject* 类型的C函数;到这里我们已经大概学习到Python如何调用C的参数和函数了.

    4.  那么 R2 上的GPIO又是如何实现被控制的呢?

        在这里大家有没有注意到,当我们运行g40.py的时候,实际上里面也是运行的C代码,再挖一层实际上当执行这个Python脚本就相当于就是执行的C代码,换句话说,就是我们只要实现了如何通过C去控制GPIO,那么就能实现Python去控制GPIO;那么我们要如何去实现C代码控制GPIO呢?

    这里插入两个知识点:用户态,内核态
    Linux系统运行的时候实际上是分内核态和用户态的,内核态和用户态之间相互分隔,但是可以通过一些特殊的接口让他们之间相互访问。
    比如我们进到Ubuntu系统,GUI,桌面软件,命令行等等这些实际上都是用户态的操作。
    一般涉及到较深层次的硬件驱动接口时,才会去操作内核态,比如我们开发嵌入式拿到的BSP代码,去写的各个硬件驱动代码一般就是属于内核部分。

        如果你使用过单片机,你可能还记得操作GPIO,直接去给对应的寄存器赋值就好了,没错思路是对的;那么我们如何通过Ubuntu系统去写芯片GPIO的寄存器呢?

        这里我直接使用mmap的办法,即内存映射,暂时就不发散用户态如何调用到内核态的方法了,后续写关于嵌入式的知识时再介绍.

        那么我们看看mmap是做什么用处的:

        简单去讲,就是虚拟地址映射到实际物理地址的方法;我们知道在单片机上面操作的寄存器地址,实际上就是它的物理地址;但是嵌入式系统不一样,嵌入式系统有一个内存管理的机制,我们叫MMU,具体MMU的原理暂时也不发散了,后续有机会再介绍(后面会专门再写一篇关于MMU的文章).

        通过查询芯片的Datasheet,我们可以找到GPIO的实际寄存器地址,然后读取出来后,就可以像单片机操作GPIO那样去写程序了,这里我贴R2的例子:

        我们要操作的GPIO口是这些:

       

    #include <stdio.h>
    #include <stdint.h>
    #include <stdlib.h>
    #include <string.h>
    #include <unistd.h>
    #include <sys/mman.h>
    #include <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    
    #define MMAP_PATH    "/dev/mem"
    
    static uint8_t* gpio_mmap_reg = NULL;
    static int gpio_mmap_fd = 0;
    
    //这里定义GPIO 原理图里面的 PIN
    int pins[] = {75, 76, 206, 80, 79, 205, 56, 55, 54, 57, 126, 74, 73, 49, 202, 82, 81, 24, 25, 21, 18, 53, 20, 58, 72, 19, 22, 200};
    //芯片GPIO寄存器
    #define GPIO_DOUT_BASE_OFFSET 0x500 #define GPIO_MODE_BASE_OFFSET 0x760 #define GPIO_REG_BASE 0x10005000 static int gpio_mmap(void) { if ((gpio_mmap_fd = open(MMAP_PATH, O_RDWR|O_SYNC)) < 0) { fprintf(stderr, "unable to open mmap file"); return -1; } gpio_mmap_reg = (uint8_t*) mmap(NULL, 1024, PROT_READ | PROT_WRITE, MAP_FILE | MAP_SHARED , gpio_mmap_fd, GPIO_REG_BASE); if (gpio_mmap_reg == MAP_FAILED) { perror("foo"); fprintf(stderr, "failed to mmap"); gpio_mmap_reg = NULL; close(gpio_mmap_fd); return -1; } return 0; } int main() { int ret = -1; int pin = 75; if (gpio_mmap()) return -1; printf("set dir "); uint32_t tmp; int position = 0; //SET MODE AS GPIO,GPIO模式的值为0 printf("BASEADDR=%X SET MODE AS GPIO ", gpio_mmap_reg); for(int i = 0; i < sizeof(pins)/sizeof(int); i++){ pin = pins[i];      position = gpio_mmap_reg + GPIO_MODE_BASE_OFFSET + (pin / 5) * 16; printf("pin=%d, positon = %X ", pin, position);     tmp = *(volatile uint32_t*)(position);     tmp &= ~(1u << ((pin % 5) * 3));    //赋值为0     *(volatile uint32_t*)(position) = tmp; } printf(" SET DIR AS OUT "); for(int i = 0; i < sizeof(pins)/sizeof(int); i++){ pin = pins[i];     if(pin < 199){ position = gpio_mmap_reg + (pin / 16) * 16; }else{ position = gpio_mmap_reg + (pin / 16) * 16 + 0x10; }   printf("pin=%d, positon = %X ", pin, position); tmp = *(volatile uint32_t*)(position);    tmp |= (1u << (pin % 16));   *(volatile uint32_t*)(position) = tmp; usleep(100000); } //SET VAULE printf(" SET VALUE AS HIGH LEVEL "); for(int i = 0; i < sizeof(pins)/sizeof(int); i++){ pin = pins[i];    position = gpio_mmap_reg + GPIO_DOUT_BASE_OFFSET + (pin / 16) * 16;    printf("pin=%d, positon = %X ", pin, position);    tmp = *(volatile uint32_t*)(position);    tmp |= (1u << (pin % 16));    *(volatile uint32_t*)(position) = tmp;    usleep(100000); }   printf(" SET VALUE AS LOW LEVEL "); for(int i = 0; i < sizeof(pins)/sizeof(int); i++){ pin = pins[i]; position = gpio_mmap_reg + GPIO_DOUT_BASE_OFFSET + (pin / 16) * 16;    printf("pin=%d, positon = %X ", pin, position); tmp = *(volatile uint32_t*)(position); printf("tmp = %X ", tmp);    tmp &= ~(1u << (pin % 16));    printf("tmp = %X ", tmp);    *(volatile uint32_t*)(position) = tmp; usleep(100000); } close(gpio_mmap_fd); }

         上面这个例子就实现了用户态操作GPIO的方法,具体地址,和运算方法跟你的芯片GPIO寄存器定义有关,看到这里如果有很多东西还是不太明白也没有关系,因为这里涉及的知识面稍微多一些,不过不要灰心,自己多实战,再回过头来看其实就变简单了,关于嵌入式的学习也欢迎大家一起留言交流.

        更多的实现请大家看github上的commit记录:https://github.com/BPI-SINOVOIP/RPi.GPIO/commits/master

  • 相关阅读:
    2020春软件工程助教工作总结 第三周
    Zend Framework MVC的结构
    Zend_Cache的使用
    小油2018 win7旗舰版64位GHOST版的,安装telnet客户端时,提示:出现错误。并非所有的功能被成功更改。
    redis常用配置参数详解
    CentOS 7 源码编译安装 Redis
    Linux(CentOS)下设置nginx开机自动启动(2个办法)
    CST,CET,UTC,GMT,DST,Unix时间戳几种常见时间概述与关系(转)
    PHP_OS的常见值
    PHP_SELF、 SCRIPT_NAME、 REQUEST_URI区别
  • 原文地址:https://www.cnblogs.com/topbin/p/10577137.html
Copyright © 2011-2022 走看看