测试程序之提供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

相关文章
|
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 自动化安装程序
209 6
Nessus Professional 10.10 Auto Installer for RHEL 10, AlmaLinux 10, Rocky Linux 10 - Nessus 自动化安装程序
|
7月前
|
测试技术 数据库 Python
解释测试中setup和teardown函数的应用。
总结起来,`setup`和 `teardown`函数就像扔宴会的主人,他们保障了宴会的流畅进行。他们是准备环境和清理现场的重要工作人员,他们的工作直接影响着我们的测试效率和质量。我们可以把 `setup`和 `teardown`想象成隐藏在幕后,默默为我们服务的工作者,他们做着我们需要但是往往忽视的工作。所以,下次当你写测试的时候,别忘了给你的 `setup`和 `teardown`留出足够的位置,因为他们的作用可能是你成功的保证。
180 14
|
安全 Linux Shell
Linux上执行内存中的脚本和程序
【9月更文挑战第3天】在 Linux 系统中,可以通过多种方式执行内存中的脚本和程序:一是使用 `eval` 命令直接执行内存中的脚本内容;二是利用管道将脚本内容传递给 `bash` 解释器执行;三是将编译好的程序复制到 `/dev/shm` 并执行。这些方法虽便捷,但也需谨慎操作以避免安全风险。
419 7
|
Linux API 开发工具
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
ijkplayer是由B站研发的移动端播放器,基于FFmpeg 3.4,支持Android和iOS。其源码托管于GitHub,截至2024年9月15日,获得了3.24万星标和0.81万分支,尽管已停止更新6年。本文档介绍了如何在Linux环境下编译ijkplayer的so库,以便在较新的开发环境中使用。首先需安装编译工具并调整/tmp分区大小,接着下载并安装Android SDK和NDK,最后下载ijkplayer源码并编译。详细步骤包括环境准备、工具安装及库编译等。更多FFmpeg开发知识可参考相关书籍。
498 0
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
|
运维 Java Linux
【运维基础知识】Linux服务器下手写启停Java程序脚本start.sh stop.sh及详细说明
### 启动Java程序脚本 `start.sh` 此脚本用于启动一个Java程序,设置JVM字符集为GBK,最大堆内存为3000M,并将程序的日志输出到`output.log`文件中,同时在后台运行。 ### 停止Java程序脚本 `stop.sh` 此脚本用于停止指定名称的服务(如`QuoteServer`),通过查找并终止该服务的Java进程,输出操作结果以确认是否成功。
893 1
|
SQL 分布式计算 Hadoop
Hadoop-14-Hive HQL学习与测试 表连接查询 HDFS数据导入导出等操作 逻辑运算 函数查询 全表查询 WHERE GROUP BY ORDER BY(一)
Hadoop-14-Hive HQL学习与测试 表连接查询 HDFS数据导入导出等操作 逻辑运算 函数查询 全表查询 WHERE GROUP BY ORDER BY(一)
254 4
|
SQL
Hadoop-14-Hive HQL学习与测试 表连接查询 HDFS数据导入导出等操作 逻辑运算 函数查询 全表查询 WHERE GROUP BY ORDER BY(二)
Hadoop-14-Hive HQL学习与测试 表连接查询 HDFS数据导入导出等操作 逻辑运算 函数查询 全表查询 WHERE GROUP BY ORDER BY(二)
210 2
|
消息中间件 分布式计算 Java
Linux环境下 java程序提交spark任务到Yarn报错
Linux环境下 java程序提交spark任务到Yarn报错
214 5
|
Linux 测试技术 芯片
在Linux中使用GPIO线【ChatGPT】
在Linux中使用GPIO线【ChatGPT】