测试程序之提供ioctl函数应用操作GPIO适用于Linux/Android

简介: 测试程序之提供ioctl函数应用操作GPIO适用于Linux/Android

当时这个测试程序是给Linux用户的示例 , 还有个对应的driver 。

程序功能介绍

这个程序是用来控制一些GPIO引脚和电源LED的。它接受两个或三个参数,分别是命令、GPIO编号和GPIO值。它会根据参数的设置,打开一个设备文件,并使用ioctl函数来执行相应的操作。它支持四种命令,分别是:

  • 0: 读取输入GPIO的值
  • 1: 设置输出GPIO的值
  • 2: 获取输出GPIO的值
  • 3: 设置电源LED的状态
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <linux/ioctl.h>
 
#define IOCTL_READ_GPIO _IOR('a', 0, int *)
#define IOCTL_SET_GPIO _IOW('a', 1, int *)
#define IOCTL_GET_GPIO _IOR('a', 2, int *)
#define IOCTL_SET_POWER_LED _IOW('a', 3, int *)
 
int main(int argc, char *argv[]) {
    int file_desc;
    int gpio_val[2];
 
    // 打开设备文件
    file_desc = open("/dev/rockchip_gpio", 0);
    if (file_desc < 0) {
        printf("无法打开设备文件: /dev/rockchip_gpio");
        exit(-1);
    }
 
    // 检查参数数量
    if (argc < 3) {
        printf("参数数量不足\n");
        exit(-1);
    }
 
    // 根据命令执行相应操作
    switch (atoi(argv[1])) {
        case 0: // 读取输入GPIO
            gpio_val[0] = atoi(argv[2]);
            if (gpio_val[0] < 1 || gpio_val[0] > 4) {
                printf("无效的输入GPIO编号: %d\n", gpio_val[0]);
                exit(-1);
            }
            gpio_val[0] = gpio_val[0] - 1;
            if (ioctl(file_desc, IOCTL_READ_GPIO, &gpio_val) == -1) {
                printf("读取GPIO失败\n");
                exit(-1);
            }
            printf("输入GPIO %d 的值为 %d\n", gpio_val[0] + 1, gpio_val[1]);
            break;
 
        case 1: // 设置输出GPIO
            if (argc < 4) {
                printf("参数数量不足\n");
                exit(-1);
            }
            gpio_val[0] = atoi(argv[2]);
            gpio_val[1] = atoi(argv[3]);
            if (gpio_val[0] < 1 || gpio_val[0] > 4) {
                printf("无效的输出GPIO编号: %d\n", gpio_val[0]);
                exit(-1);
            }
            gpio_val[0] = gpio_val[0] - 1;
            if (ioctl(file_desc, IOCTL_SET_GPIO, &gpio_val) == -1) {
                printf("设置GPIO失败\n");
                exit(-1);
            }
            printf("设置输出GPIO %d 的值为 %d\n", gpio_val[0] + 1, gpio_val[1]);
            break;
 
        case 2: // 获取输出GPIO的值
            gpio_val[0] = atoi(argv[2]);
            if (gpio_val[0] < 1 || gpio_val[0] > 4) {
                printf("无效的输出GPIO编号: %d\n", gpio_val[0]);
                exit(-1);
            }
            gpio_val[0] = gpio_val[0] - 1;
            if (ioctl(file_desc, IOCTL_GET_GPIO, &gpio_val) == -1) {
                printf("获取GPIO值失败\n");
                exit(-1);
            }
            printf("输出GPIO %d 的值为 %d\n", gpio_val[0] + 1, gpio_val[1]);
            break;
 
        case 3: // 设置电源LED的状态
            gpio_val[0] = atoi(argv[2]);
            if (ioctl(file_desc, IOCTL_SET_POWER_LED, &gpio_val) == -1) {
                printf("设置电源LED失败\n");
                exit(-1);
            }
            printf("设置电源LED的状态为 %d\n", gpio_val[0]);
            break;
 
        default:
            printf("无效的命令\n");
    }
 
    // 关闭设备文件
    close(file_desc);
 
    return 0;
}

使用示例

要编译这个程序,需要在终端中输入:

./gcc-linaro-6.3.1-2017.05-x86_64_aarch64-linux-gnu/bin/aarch64-linux-gnu-gcc  -o gpio gpio.c

要运行这个程序,需要在终端中输入:

./gpio [command] [gpio_num] [gpio_val]

其中:

  • command: 命令编号,0~3之间的整数
  • gpio_num: GPIO编号,1~4之间的整数
  • gpio_val: GPIO值,0或1

例如:

./gpio 0 1 # 读取输入GPIO 1 的值

./gpio 1 2 1 # 设置输出GPIO 2 的值为 1

./gpio 2 3 # 获取输出GPIO 3 的值

./gpio 3 0 # 设置电源LED的状态为 0


然后就可以看到类似下面的输出

输入GPIO 1 的值为 0

设置输出GPIO 2 的值为 1

输出GPIO 3 的值为 0

设置电源LED的状态为 0

相关文章
|
4月前
|
监控 安全 Shell
管道符在渗透测试与网络安全中的全面应用指南
管道符是渗透测试与网络安全中的关键工具,既可用于高效系统管理,也可能被攻击者利用实施命令注入、权限提升、数据外泄等攻击。本文全面解析管道符的基础原理、实战应用与防御策略,涵盖Windows与Linux系统差异、攻击技术示例及检测手段,帮助安全人员掌握其利用方式与防护措施,提升系统安全性。
197 6
|
2月前
|
安全 Linux iOS开发
Nessus Professional 10.10 Auto Installer for RHEL 10, AlmaLinux 10, Rocky Linux 10 - Nessus 自动化安装程序
Nessus Professional 10.10 Auto Installer for RHEL 10, AlmaLinux 10, Rocky Linux 10 - Nessus 自动化安装程序
157 6
Nessus Professional 10.10 Auto Installer for RHEL 10, AlmaLinux 10, Rocky Linux 10 - Nessus 自动化安装程序
|
3月前
|
人工智能 数据可视化 测试技术
AI 时代 API 自动化测试实战:Postman 断言的核心技巧与实战应用
AI 时代 API 自动化测试实战:Postman 断言的核心技巧与实战应用
476 11
|
4月前
|
机器学习/深度学习 存储 分布式计算
Java 大视界 --Java 大数据机器学习模型在金融风险压力测试中的应用与验证(211)
本文探讨了Java大数据与机器学习模型在金融风险压力测试中的创新应用。通过多源数据采集、模型构建与优化,结合随机森林、LSTM等算法,实现信用风险动态评估、市场极端场景模拟与操作风险预警。案例分析展示了花旗银行与蚂蚁集团的智能风控实践,验证了技术在提升风险识别效率与降低金融风险损失方面的显著成效。
|
4月前
|
人工智能 IDE 测试技术
Browser-Use在UI自动化测试中的应用
Browser-Use是一款浏览器自动化工具,具备视觉与HTML解析、多标签管理、操作记录与复现、自定义操作、自我纠正及并行执行等功能,助力AI智能体高效完成网页任务。
329 0
|
7月前
|
测试技术 数据库 Python
解释测试中setup和teardown函数的应用。
总结起来,`setup`和 `teardown`函数就像扔宴会的主人,他们保障了宴会的流畅进行。他们是准备环境和清理现场的重要工作人员,他们的工作直接影响着我们的测试效率和质量。我们可以把 `setup`和 `teardown`想象成隐藏在幕后,默默为我们服务的工作者,他们做着我们需要但是往往忽视的工作。所以,下次当你写测试的时候,别忘了给你的 `setup`和 `teardown`留出足够的位置,因为他们的作用可能是你成功的保证。
159 14
|
监控 Android开发
【Android 逆向】函数拦截 ( GOT 表拦截 与 插桩拦截 | 插桩拦截简介 | 插桩拦截涉及的 ARM 和 x86 中的跳转指令 )
【Android 逆向】函数拦截 ( GOT 表拦截 与 插桩拦截 | 插桩拦截简介 | 插桩拦截涉及的 ARM 和 x86 中的跳转指令 )
264 0
【Android 逆向】函数拦截 ( GOT 表拦截 与 插桩拦截 | 插桩拦截简介 | 插桩拦截涉及的 ARM 和 x86 中的跳转指令 )
|
存储 Android开发
【Android 逆向】函数拦截 ( GOT 表数据结构分析 | 函数根据 GOT 表进行跳转的流程 )
【Android 逆向】函数拦截 ( GOT 表数据结构分析 | 函数根据 GOT 表进行跳转的流程 )
219 0
【Android 逆向】函数拦截 ( GOT 表数据结构分析 | 函数根据 GOT 表进行跳转的流程 )
|
存储 缓存 Java
【Android 逆向】函数拦截 ( 使用 cache_flush 系统函数刷新 CPU 高速缓存 | 刷新 CPU 高速缓存弊端 | 函数拦截推荐时机 )
【Android 逆向】函数拦截 ( 使用 cache_flush 系统函数刷新 CPU 高速缓存 | 刷新 CPU 高速缓存弊端 | 函数拦截推荐时机 )
251 0
【Android 逆向】函数拦截 ( 使用 cache_flush 系统函数刷新 CPU 高速缓存 | 刷新 CPU 高速缓存弊端 | 函数拦截推荐时机 )