Android系统APK操作GPIO读写及添加权限

需求描述

上层APK操作GPIO的diretction、value节点,实现点亮和熄灭灯,使用JNI方式,使用以下方式无效

extern "C"
JNIEXPORT jint JNICALL
Java_com_example_administrator_jni_Gpio_setOpenLed(JNIEnv *env, jclass instance) {
    int sd;
    system("echo 9 > /sys/class/gpio/export");
    system("echo out > /sys/class/gpio/gpio9/direction");
    system("echo 1 > /sys/class/gpio/gpio9/value");
    system("echo 9 > /sys/class/gpio/unexport");
    return 0;
}
实现方案
  • GPIOControl.c:JNI调用接口
#include <jni.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <unistd.h>
#include <android/log.h>

#define TAG "jni_gpio"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG ,__VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,TAG ,__VA_ARGS__)
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,TAG ,__VA_ARGS__)
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,TAG ,__VA_ARGS__)
#define LOGF(...) __android_log_print(ANDROID_LOG_FATAL,TAG ,__VA_ARGS__)

#define IN              0
#define OUT             1
#define LOW             0
#define HIGH            1

#define BUFFER_MAX    3
#define DIRECTION_MAX 48

/*
 * Class:     com_example_jnigpio_GPIOControl
 * Method:    exportGpio
 * Signature: (I)I
 */
JNIEXPORT jint JNICALL Java_com_example_jnigpio_GPIOControl_exportGpio(JNIEnv *env, jobject instance, jint gpio)
{
    char buffer[BUFFER_MAX];
    int len;
    int fd;

    fd = open("/sys/class/gpio/export", O_WRONLY);
    if (fd < 0) {
        LOGE("Failed to open export for writing!\n");
        return(0);
    }

    len = snprintf(buffer, BUFFER_MAX, "%d", gpio);
    if (write(fd, buffer, len) < 0) {
        LOGE("Fail to export gpio!\n");
        return 0;
    }

    close(fd);
    return 1;
}

/*
 * Class:     com_example_jnigpio_GPIOControl
 * Method:    setGpioDirection
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_example_jnigpio_GPIOControl_setGpioDirection(JNIEnv *env, jobject instance, jint gpio, jint direction)
{
    static const char dir_str[]  = "in\0out";
    char path[DIRECTION_MAX];
    int fd;

    snprintf(path, DIRECTION_MAX, "/sys/class/gpio/gpio%d/direction", gpio);
    fd = open(path, O_WRONLY);
    if (fd < 0) {
        LOGE("failed to open gpio direction for writing!\n");
        return 0;
    }

    if (write(fd, &dir_str[direction == IN ? 0 : 3], direction == IN ? 2 : 3) < 0) {
        LOGE("failed to set direction!\n");
        return 0;
    }

    close(fd);
    return 1;
}

/*
 * Class:     com_example_jnigpio_GPIOControl
 * Method:    readGpioStatus
 * Signature: (I)I
 */
JNIEXPORT jint JNICALL Java_com_example_jnigpio_GPIOControl_readGpioStatus(JNIEnv *env, jobject instance, jint gpio)
{
    char path[DIRECTION_MAX];
    char value_str[3];
    int fd;

    snprintf(path, DIRECTION_MAX, "/sys/class/gpio/gpio%d/value", gpio);
    fd = open(path, O_RDONLY);
    if (fd < 0) {
        LOGE("failed to open gpio value for reading!\n");
        return -1;
    }

    if (read(fd, value_str, 3) < 0) {
        LOGE("failed to read value!\n");
        return -1;
    }

    close(fd);
    return (atoi(value_str));
}

/*
 * Class:     com_example_jnigpio_GPIOControl
 * Method:    writeGpioStatus
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_example_jnigpio_GPIOControl_writeGpioStatus(JNIEnv *env, jobject instance, jint gpio, jint value)
{
    static const char values_str[] = "01";
    char path[DIRECTION_MAX];
    int fd;

    snprintf(path, DIRECTION_MAX, "/sys/class/gpio/gpio%d/value", gpio);
    fd = open(path, O_WRONLY);
    if (fd < 0) {
        LOGE("failed to open gpio value for writing!\n");
        return 0;
    }

    if (write(fd, &values_str[value == LOW ? 0 : 1], 1) < 0) {
        LOGE("failed to write value!\n");
        return 0;
    }

    close(fd);
    return 1;
}

/*
 * Class:     com_example_jnigpio_GPIOControl
 * Method:    unexportGpio
 * Signature: (I)I
 */
JNIEXPORT jint JNICALL Java_com_example_jnigpio_GPIOControl_unexportGpio(JNIEnv *env, jobject instance, jint gpio)
{
    char buffer[BUFFER_MAX];
    int len;
    int fd;

    fd = open("/sys/class/gpio/unexport", O_WRONLY);
    if (fd < 0) {
        LOGE("Failed to open unexport for writing!\n");
        return 0;
    }

    len = snprintf(buffer, BUFFER_MAX, "%d", gpio);
    if (write(fd, buffer, len) < 0) {
        LOGE("Fail to unexport gpio!");
        return 0;
    }

    close(fd);
    return 1;
}

  • Android.mk:编译出libGPIOControl.so
LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE    := GPIOControl
LOCAL_SRC_FILES := GPIOControl.c
LOCAL_LDLIBS    := -llog

include $(BUILD_SHARED_LIBRARY)
  • 添加节点读写权限
device/qcom/common/rootdir/etc/init.qcom.rc 

 on boot
+    chmod 0777 /sys/class/gpio/export
+    chmod 0777 /sys/class/gpio/unexport
  • 添加实际节点的读写权限:ls -al sys/class/gpio/查看实际节点
device/qcom/common/rootdir/etc/ueventd.qcom.rc
 
+/sys/devices/soc/1000000.pinctrl/gpio/gpio* direction 0777 system system
+/sys/devices/soc/1000000.pinctrl/gpio/gpio* value 0777 system system
  • 添加节点的SELinux权限
device/qcom/sepolicy/common/system_app.te

+allow system_app sysfs:file rw_file_perms;
  • APK调用
  1. Android应用修改AndroidManifest.xml文件,在manifest节点中添加android:sharedUserId="android.uid.system,修改Android.mk文件,添加LOCAL_CERTIFICATE := platform设置系统签名。
  2. Android APK调用JNI加载so动态库:
    https://blog.csdn.net/weixin_44008788/article/details/114393087
相关拓展
  • 节点读写
//读节点
private static String readSys(String path) {
    String prop = "waiting";// 默认值
    try {
        BufferedReader reader = new BufferedReader(new FileReader(path));
        prop = reader.readLine();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return prop;
}
//调用方法
 readSys("/sys/class/demo/version")
//写节点
private static String writeSys(String path) {
    try {
        BufferedWriter bufWriter = null;
        bufWriter = new BufferedWriter(new FileWriter(path));
        bufWriter.write("1");  // 写操作
        bufWriter.close();
    } catch (IOException e) {
        e.printStackTrace();
    }
}
//调用方法
writeSys("/sys/class/demo/version")
//以FileOutputStream写节点
void IOCtrl(int pin, int level) {
    String path;
    path = "/sys/class/gpio/gpio" + pin + "/value";
    try {
        FileOutputStream out = new FileOutputStream(path);
        out.write(level);
        out.flush();
        out.close();
    } catch (IOException e) {
        e.printStackTrace();
        return;
    }
}
  • 添加设备文件节点权限

/sys/class/leds/green/brightness //快捷方式
/sys/devices/soc.0/gpio-leds.66/leds/green/brightness //实际节点

操作LED灯的设备文件节点为APP层system app进程开放该节点访问权限(读或写),权限配置主要修改/device/qcom/sepolicy/common目录下的file.te、file_contexts和system_app.te三个文件

file.te修改如下

# GPIO accessed by system app
type sysfs_gpio, fs_type, sysfs_type;

file_contexts修改如下

/sys/devices/soc/1010000.pinctrl/gpio/gpio62/value           u:object_r:sysfs_gpio:s0
/sys/devices/soc/1010000.pinctrl/gpio/gpio63/value           u:object_r:sysfs_gpio:s0

system_app.te修改如下

allow system_app sysfs_gpio:file rw_file_perms;  

如果通过以上添加SELinux之后,仍没有权限读写sys或proc节点,需要到init.rc里面配置如下

chown system system 文件结点
chmod 777 文件结点
相关参考

https://blog.csdn.net/wds1181977/article/details/54967110
https://blog.csdn.net/liangtianmeng/article/details/84698568
https://blog.csdn.net/d38825/article/details/81240851
https://blog.csdn.net/K_Hello/article/details/90038078
android 查看gpio状态_Android应用层操作GPIO
Android app如何正确读写系统sys设备节点

<think>好的,我现在要解决用户关于Android读写节点权限的问题。首先,我需要理解用户的问题到底是什么。用户提到的是“Android读写节点权限问题”,可能指的是在Android系统中访问硬件节点(比如/sys或/dev下的文件)时遇到的权限限制。这类问题通常出现在需要与硬件交互的应用程序开发中,比如控制LED、调整背光亮度等。 接下来,我需要回忆Android系统权限机制。Android基于Linux,所以文件系统权限模型和Linux类似,每个文件都有所有者、组和其他用户的读、写、执行权限。默认情况下,普通应用没有权限访问系统级的节点,因为这些节点通常属于root用户或系统组,权限设置为660(rw-rw----)或者更严格。用户的应用如果尝试访问这些节点,可能会遇到“Permission denied”的错误。 然后,我需要考虑用户可能的场景。用户可能是在开发一个需要直接与硬件交互的App,比如调整屏幕亮度、控制GPIO等。这时候,如何让应用获得足够的权限读写这些节点呢?常见的解决方法有几种: 1. 修改文件权限:通过chmod或chown改变节点的权限或所有者,但这需要root权限,普通应用无法直接操作,可能需要在设备root后处理,或者让用户手动修改,但这对普通用户来说不友好。 2. 使用系统权限:有些节点属于特定的组,比如sysfs组。如果应用申请了相应的系统权限,可能被允许访问。例如,在AndroidManifest.xml中添加android:sharedUserId="android.uid.system",然后使用平台签名签名应用。但这种方法需要应用成为系统应用,通常只有设备制造商或ROM开发者才能做到。 3. 使用JNI和su命令:通过调用su来执行命令,但这需要设备已经root,并且用户授权超级用户权限。不过,这种方式安全风险大,且不适用于非root设备。 4. 使用Android框架API:比如通过Settings.System来调整亮度,而不是直接写节点。这是推荐的方式,因为它不涉及底层权限问题,但可能无法覆盖所有硬件节点的需求。 然后,我需要考虑如何逐步引导用户解决问题。首先确认用户是否真的需要直接访问节点,是否有更高层次的API可用。如果必须访问节点,检查节点的权限和所属组,看是否有系统权限可以申请。如果需要系统权限,可能需要将应用内置为系统应用,或者与设备制造商合作。 另外,可能需要考虑SELinux的影响。在较新的Android版本中,SELinux会限制即使有文件权限的应用访问节点,因此还需要调整SELinux策略,这通常需要修改系统镜像,普通开发者难以实现。 还有,用户可能需要了解如何在代码中正确打开和读写节点文件,比如使用FileInputStream或FileOutputStream,并处理可能的异常,比如SecurityException或IOException。 最后,总结一下,访问硬件节点通常需要系统权限或设备root,普通应用很难直接实现,建议优先使用官方API,如果必须访问节点,可能需要系统签名或root权限,同时处理SELinux策略问题。</think>在Android系统中,读写硬件节点(如`/sys`或`dev`目录下的文件)涉及系统权限控制。以下是逐步分析和解决方案: --- ### **1. 问题背景** - **硬件节点**:Android基于Linux内核,硬件设备通过虚拟文件(如`/sys/class/leds/backlight/brightness`)暴露给用户层。 - **权限限制**:默认情况下,普通应用无权限读写这些节点(权限通常为`660`,仅允许`root`或特定系统组访问)。 --- ### **2. 常见错误** ```bash # 尝试直接写入节点时可能报错: adb shell "echo 100 > /sys/class/leds/backlight/brightness" # 错误信息:Permission denied ``` --- ### **3. 解决方案** #### **方案1:修改文件权限(需Root)** - **步骤**: 1. 通过ADB或终端修改节点权限: ```bash adb root adb shell chmod 666 /sys/class/leds/backlight/brightness ``` 2. 应用直接通过`FileOutputStream`读写文件。 - **缺点**:需Root权限,重启后失效。 #### **方案2:申请系统权限(需系统签名)** - **适用场景**:设备厂商或系统应用开发。 - **步骤**: 1. 在`AndroidManifest.xml`声明系统UID: ```xml android:sharedUserId="android.uid.system" ``` 2. 使用**平台签名**编译APK。 3. 通过标准Java IO读写节点。 - **缺点**:需与设备厂商合作或刷入自定义系统。 #### **方案3:通过JNI调用底层命令(需Root)** - **步骤**: 1. 在Native代码中执行`su`命令: ```c system("echo 100 > /sys/class/leds/backlight/brightness"); ``` 2. 在Java中调用JNI函数。 - **缺点**:需用户授权Root,兼容性差。 #### **方案4:使用Android框架API** - **推荐方式**:优先使用官方API操作硬件。 ```java // 调整屏幕亮度(无需写节点) Settings.System.putInt( getContentResolver(), Settings.System.SCREEN_BRIGHTNESS, 100 ); ``` - **优点**:无需处理权限问题。 --- ### **4. SELinux策略限制** - **问题**:即使文件权限开放,SELinux可能阻止访问。 - **解决方案**: 1. 修改SELinux规则(需系统镜像权限): ```te # 在device/xxx/sepolicy/file_contexts中添加: /sys/class/leds/backlight/brightness u:object_r:sysfs:s0 ``` 2. 添加允许访问的SELinux规则。 --- ### **5. 代码示例(读写节点)** ```java try { File file = new File("/sys/class/leds/backlight/brightness"); FileOutputStream fos = new FileOutputStream(file); fos.write("100".getBytes()); fos.close(); } catch (IOException e) { e.printStackTrace(); } ``` --- ### **6. 总结** | 方法 | 必要条件 | 稳定性 | 适用范围 | |-------------------|-----------------------|--------|------------------| | 修改文件权限 | Root权限 | 低 | 测试环境 | | 系统签名 | 平台签名 | 高 | 设备厂商 | | JNI+Root | Root权限 | 中 | 调试用途 | | Android API | 无特殊权限 | 高 | 通用应用开发 | **建议**:优先通过Android官方API实现功能,若必须操作节点,需与设备厂商合作获取系统权限
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值