您当前的位置: 首页 > 

韦东山

暂无认证

  • 0浏览

    0关注

    506博文

    0收益

  • 0浏览

    0点赞

    0打赏

    0留言

私信
关注
热门博文

08_调试与使用虚拟的GPIO控制器

韦东山 发布时间:2021-08-26 10:14:59 ,浏览量:0

目录
    • 资料下载
    • 视频观看
    • 调试与使用虚拟的GPIO控制器
      • 1. 硬件功能
      • 2. 编写设备树文件
      • 3. 上机实验
        • 3.2 编译、替换设备树
        • 3.3 编译、安装驱动程序
      • 4. STM32MP157上的bug

百问网技术交流群,百万嵌入式工程师聚集地: https://www.100ask.net/page/2248041

资料下载

coding无法使用浏览器打开,必须用git工具下载:

git clone https://e.coding.net/weidongshan/linux/doc_and_source_for_drivers.git
视频观看

百问网驱动大全

调试与使用虚拟的GPIO控制器

参考资料:

  • Linux 5.x内核文档

    • Linux-5.4\Documentation\driver-api
    • Linux-5.4\Documentation\devicetree\bindings\gpio\gpio.txt
    • Linux-5.4\drivers\gpio\gpio-74x164.c
  • Linux 4.x内核文档

    • Linux-4.9.88\Documentation\gpio
    • Linux-4.9.88\Documentation\devicetree\bindings\gpio\gpio.txt
    • Linux-4.9.88\drivers\gpio\gpio-74x164.c
  • 本章课程源码位于GIT仓库里

    doc_and_source_for_drivers\IMX6ULL\source\07_GPIO\03_virtual_gpio_ok
    doc_and_source_for_drivers\STM32MP157\source\A7\07_GPIO\03_virtual_gpio_ok
    
1. 硬件功能

假设使用这个虚拟的GPIO Controller的pinA来控制LED:

在这里插入图片描述

2. 编写设备树文件
gpio_virt: virtual_gpiocontroller {
	compatible = "100ask,virtual_gpio";
    gpio-controller;
    #gpio-cells = ;
    ngpios = ;
};

myled {
	compatible = "100ask,leddrv";
	led-gpios = ;
};
3. 上机实验

3.1 设置工具链

1. STM32MP157 注意:对于STM32MP157,以前说编译内核/驱动、编译APP的工具链不一样,其实编译APP用的工具链也能用来编译内核。

export ARCH=arm
export CROSS_COMPILE=arm-buildroot-linux-gnueabihf-
export PATH=$PATH:/home/book/100ask_stm32mp157_pro-sdk/ToolChain/arm-buildroot-linux-gnueabihf_sdk-buildroot/bin

2. IMX6ULL

export ARCH=arm
export CROSS_COMPILE=arm-linux-gnueabihf-
export PATH=$PATH:/home/book/100ask_imx6ull-sdk/ToolChain/gcc-linaro-6.2.1-2016.11-x86_64_arm-linux-gnueabihf/bin
3.2 编译、替换设备树

1. STM32MP157

  • 修改arch/arm/boot/dts/stm32mp157c-100ask-512d-lcd-v1.dts,添加如下代码:

    / {
        gpio_virt: virtual_gpiocontroller {
            compatible = "100ask,virtual_gpio";
            gpio-controller;
            #gpio-cells = ;
            ngpios = ;
        };
    
        myled {
            compatible = "100ask,leddrv";
            led-gpios = ;
        };
    };
    
  • 编译设备树: 在Ubuntu的STM32MP157内核目录下执行如下命令, 得到设备树文件:arch/arm/boot/dts/stm32mp157c-100ask-512d-lcd-v1.dtb

    make dtbs
    
  • 复制到NFS目录:

    $ cp arch/arm/boot/dts/stm32mp157c-100ask-512d-lcd-v1.dtb ~/nfs_rootfs/
    
  • 开发板上挂载NFS文件系统

    • vmware使用NAT(假设windowsIP为192.168.1.100)

      [root@100ask:~]# mount -t nfs -o nolock,vers=3,port=2049,mountport=9999 
      192.168.1.100:/home/book/nfs_rootfs /mnt
      
    • vmware使用桥接,或者不使用vmware而是直接使用服务器:假设Ubuntu IP为192.168.1.137

      [root@100ask:~]#  mount -t nfs -o nolock,vers=3 192.168.1.137:/home/book/nfs_rootfs /mnt
      
  • 确定设备树分区挂载在哪里

    由于版本变化,STM32MP157单板上烧录的系统可能有细微差别。 在开发板上执行cat /proc/mounts后,可以得到两种结果(见下图):

    • mmcblk2p2分区挂载在/boot目录下(下图左边):无需特殊操作,下面把文件复制到/boot目录即可

    • mmcblk2p2挂载在/mnt目录下(下图右边)

      • 在视频里、后面文档里,都是更新/boot目录下的文件,所以要先执行以下命令重新挂载:
        • mount /dev/mmcblk2p2 /boot

在这里插入图片描述

  • 更新设备树

    [root@100ask:~]# cp /mnt/stm32mp157c-100ask-512d-lcd-v1.dtb /boot
    [root@100ask:~]# sync
    
  • 重启开发板

2. IMX6ULL

  • 修改arch/arm/boot/dts/100ask_imx6ull-14x14.dts,添加如下代码:

    / {
        gpio_virt: virtual_gpiocontroller {
            compatible = "100ask,virtual_gpio";
            gpio-controller;
            #gpio-cells = ;
            ngpios = ;
        };
    
        myled {
            compatible = "100ask,leddrv";
            led-gpios = ;
        };
    };
    
  • 编译设备树: 在Ubuntu的IMX6ULL内核目录下执行如下命令, 得到设备树文件:arch/arm/boot/dts/100ask_imx6ull-14x14.dtb

    make dtbs
    
  • 复制到NFS目录:

    $ cp arch/arm/boot/dts/100ask_imx6ull-14x14.dtb ~/nfs_rootfs/
    
  • 开发板上挂载NFS文件系统

    • vmware使用NAT(假设windowsIP为192.168.1.100)

      [root@100ask:~]# mount -t nfs -o nolock,vers=3,port=2049,mountport=9999 
      192.168.1.100:/home/book/nfs_rootfs /mnt
      
    • vmware使用桥接,或者不使用vmware而是直接使用服务器:假设Ubuntu IP为192.168.1.137

      [root@100ask:~]#  mount -t nfs -o nolock,vers=3 192.168.1.137:/home/book/nfs_rootfs /mnt
      
    • 更新设备树

      [root@100ask:~]# cp /mnt/100ask_imx6ull-14x14.dtb /boot
      [root@100ask:~]# sync
      
  • 重启开发板

3.3 编译、安装驱动程序
  • 编译:

    • 在Ubuntu上
    • 修改01_led02_virtual_gpio中的Makefile,指定内核路径KERN_DIR,在执行make命令即可。
  • 安装:

    • 在开发板上

    • 挂载NFS,复制文件,insmod,类似如下命令:

      mount -t nfs -o nolock,vers=3 192.168.1.137:/home/book/nfs_rootfs /mnt
      // 对于IMX6ULL,想看到驱动打印信息,需要先执行
      echo "7 4 1 7" > /proc/sys/kernel/printk
      
      insmod -f /mnt/virtual_gpio_driver.ko
      insmod -f /mnt/leddrv.ko
      
      ls /dev/100ask_led0
      /mnt/ledtest /dev/100ask_led0 on
      /mnt/ledtest /dev/100ask_led0 off
      
  • 观察内核打印的信息

4. STM32MP157上的bug

在STM32MP157上做如下实验时:

echo 509 > /sys/class/gpio/export
echo out > /sys/class/gpio/gpio509/direction
cat /sys/class/gpio/gpio509/value

发现对于value执行一次cat操作,导致virt_gpio_get_value函数被调用3次,如下:

cat /sys/class/gpio/gpio509/value
[   96.283263] get pin 1, it's val = 0
[   96.297803] get pin 1, it's val = 0
[   96.312604] get pin 1, it's val = 0

cat value这个操作,会导致驱动drivers/gpio/gpiolib-sysfs.c中的value_show函数被调用。

value_show只会调用一次GPIO Controller中的get函数。

所以,我们编写了一个read.c程序,源码如下:

#include 
#include 
#include 
#include 
#include 

int main(int argc, char **argv)
{
        int fd = open(argv[1], O_RDONLY);
    
        char buf[10];
    
        read(fd, buf, 10);
        printf("%s\n", buf);
        return 0;
}

编译read.c:

arm-buildroot-linux-gnueabihf-gcc -o read read.c

放到板子上执行,发现读取value文件一次,只会导致get函数被调用一次,如下:

# ./read /sys/class/gpio/gpio509/value
[  298.663613] get pin 1, it's val = 0
1

所以:问题在于cat命令,虽然我们执行了一次cat操作,但是它发起了3次读value文件的操作。

  return 0;

}


编译read.c:

```shell
arm-buildroot-linux-gnueabihf-gcc -o read read.c

放到板子上执行,发现读取value文件一次,只会导致get函数被调用一次,如下:

# ./read /sys/class/gpio/gpio509/value
[  298.663613] get pin 1, it's val = 0
1

所以:问题在于cat命令,虽然我们执行了一次cat操作,但是它发起了3次读value文件的操作。

至于cat的bug在哪,无关紧要,先不花时间去查。

百问网技术交流群,百万嵌入式工程师聚集地: https://www.100ask.net/page/2248041

关注
打赏
1658827356
查看更多评论
立即登录/注册

微信扫码登录

0.0420s