Android从硬件到APP:增加硬件服务

Android从硬件到APP:增加硬件服务

Android平台应用越来越广泛,在一些应用领域我们经常需要增加一些硬件控制等特殊功能,而标准Android框架仅支持手机、平板等常见外围器件,比如:相机、震动、电话、蓝牙、音频等。下面就以某个项目需要实现APP操作GPIO来讲述开发的整个过程。

驱动

  1. 在kernel/drivers/misc/目录下新建topband_gpio目录,并增加下面几个文件:
    topband_gpio.c
#include <linux/module.h>
#include <linux/init.h>
#include <linux/device.h>
#include <linux/errno.h>
#include <linux/err.h>
#include <linux/kernel.h>
#include <linux/ctype.h>
#include <linux/delay.h>
#include <linux/idr.h>
#include <linux/sched.h>
#include <linux/slab.h>
#include <linux/interrupt.h>
#include <linux/signal.h>
#include <linux/pm.h>
#include <linux/notifier.h>
#include <linux/fb.h>
#include <linux/input.h>
#include <linux/ioport.h>
#include <linux/io.h>
#include <linux/clk.h>
#include <linux/pinctrl/consumer.h>
#include <linux/platform_device.h>
#include <linux/kthread.h>
#include <linux/time.h>
#include <linux/timer.h>
#include <linux/regulator/consumer.h>
#include <linux/gpio.h>
#include <linux/of.h>
#include <linux/of_irq.h>
#include <linux/of_gpio.h>
#include <linux/of_platform.h>
#include <linux/miscdevice.h>
#include <linux/uaccess.h>

#define TOPBAND_GPIO_NAME "topband,gpio"

// ioctl cmd
#define TOPBAND_GPIO_IOC_MAGIC  'f'

#define TOPBAND_GPIO_IOC_SET_VALUE _IOW(TOPBAND_GPIO_IOC_MAGIC, 1, int)

#define TOPBAND_GPIO_IOC_MAXNR 1

struct topband_gpio_data {
    struct platform_device  *platform_dev;
    struct miscdevice topband_gpio_device;
    int                     gpio1;
    int                     gpio2;
    int                     gpio3;
    int                     gpio4;
};

static void topband_gpio_free_io_port(struct topband_gpio_data *topband_gpio)
{
    if(gpio_is_valid(topband_gpio->gpio1)) {
        gpio_free(topband_gpio->gpio1);
    }
    if(gpio_is_valid(topband_gpio->gpio2)) {
        gpio_free(topband_gpio->gpio2);
    }
    if(gpio_is_valid(topband_gpio->gpio3)) {
        gpio_free(topband_gpio->gpio3);
    }
    if(gpio_is_valid(topband_gpio->gpio4)) {
        gpio_free(topband_gpio->gpio4);
    }
    return;
}

static int topband_gpio_parse_dt(struct device *dev,
                              struct topband_gpio_data *topband_gpio)
{
    struct device_node *np = dev->of_node;

    topband_gpio->gpio1 = of_get_named_gpio(np, "gpio-1", 0);
    if(!gpio_is_valid(topband_gpio->gpio1)) {
        dev_err(dev, "No valid gpio1");
    }

    topband_gpio->gpio2 = of_get_named_gpio(np, "gpio-2", 0);
    if(!gpio_is_valid(topband_gpio->gpio2)) {
        dev_err(dev, "No valid gpio2");
    }

    topband_gpio->gpio3 = of_get_named_gpio(np, "gpio-3", 0);
    if(!gpio_is_valid(topband_gpio->gpio3)) {
        dev_err(dev, "No valid gpio3");
    }

    topband_gpio->gpio4 = of_get_named_gpio(np, "gpio-4", 0);
    if(!gpio_is_valid(topband_gpio->gpio4)) {
        dev_err(dev, "No valid gpio4");
    }

    return 0;
}

static int topband_gpio_request_io_port(struct topband_gpio_data *topband_gpio)
{
    int ret = 0;

    if(gpio_is_valid(topband_gpio->gpio1)) {
        ret = gpio_request(topband_gpio->gpio1, "topband_gpio_1");

        if(ret < 0) {
            dev_err(&topband_gpio->platform_dev->dev,
                    "Failed to request GPIO1:%d, ERRNO:%d\n",
                    (s32)topband_gpio->gpio1, ret);
            return -ENODEV;
        }

        gpio_direction_output(topband_gpio->gpio1, 0);
        dev_info(&topband_gpio->platform_dev->dev, "Success request gpio1\n");
    }

    if(gpio_is_valid(topband_gpio->gpio2)) {
        ret = gpio_request(topband_gpio->gpio2, "topband_gpio_2");

        if(ret < 0) {
            dev_err(&topband_gpio->platform_dev->dev,
                    "Failed to request GPIO2:%d, ERRNO:%d\n",
                    (s32)topband_gpio->gpio2, ret);
            return -ENODEV;
        }

        gpio_direction_output(topband_gpio->gpio2, 0);
        dev_info(&topband_gpio->platform_dev->dev, "Success request gpio2\n");
    }

    if(gpio_is_valid(topband_gpio->gpio3)) {
        ret = gpio_request(topband_gpio->gpio3, "topband_gpio_3");

        if(ret < 0) {
            dev_err(&topband_gpio->platform_dev->dev,
                    "Failed to request GPIO3:%d, ERRNO:%d\n",
                    (s32)topband_gpio->gpio3, ret);
            return -ENODEV;
        }

        gpio_direction_output(topband_gpio->gpio3, 0);
        dev_info(&topband_gpio->platform_dev->dev, "Success request gpio3\n");
    }

    if(gpio_is_valid(topband_gpio->gpio4)) {
        ret = gpio_request(topband_gpio->gpio4, "topband_gpio_4");

        if(ret < 0) {
            dev_err(&topband_gpio->platform_dev->dev,
                    "Failed to request GPIO4:%d, ERRNO:%d\n",
                    (s32)topband_gpio->gpio4, ret);
            return -ENODEV;
        }

        gpio_direction_output(topband_gpio->gpio4, 0);
        dev_info(&topband_gpio->platform_dev->dev, "Success request gpio4\n");
    }

    return ret;
}

static int topband_gpio_set_value(int gpio, int value) {
    if(gpio_is_valid(gpio)) {
        gpio_set_value(gpio, value);
        return 0;
    }
    return -1;
}

static int topband_gpio_ctrl(struct topband_gpio_data *topband_gpio, int data) {
    int ret = 0;
    int gpio, value;

    gpio = (data >> 4) & 0x0f;
    value = data & 0x0f;

    dev_info(&topband_gpio->platform_dev->dev, 
                    "%s, gpio=%d, value=%d\n", __func__, gpio, value);
    switch (gpio) {
        case 1:
            ret = topband_gpio_set_value(topband_gpio->gpio1, value);
            break;
        case 2:
            ret = topband_gpio_set_value(topband_gpio->gpio2, value);
            break;
        case 3:
            ret = topband_gpio_set_value(topband_gpio->gpio3, value);
            break;
        case 4:
            ret = topband_gpio_set_value(topband_gpio->gpio4, value);
            break;
        default:
            ret = -1;
            break;
    }
    return ret;
}

static int topband_gpio_dev_open(struct inode *inode, struct file *filp)
{
    int ret = 0;

    struct topband_gpio_data *topband_gpio = container_of(filp->private_data,
                               struct topband_gpio_data,
                               topband_gpio_device);
    filp->private_data = topband_gpio;
    dev_info(&topband_gpio->platform_dev->dev,
         "device node major=%d, minor=%d\n", imajor(inode), iminor(inode));

    return ret;
}

static long topband_gpio_dev_ioctl(struct file *pfile,
                     unsigned int cmd, unsigned long arg)
{
    int ret = 0;
    int data = 0;
    struct topband_gpio_data *topband_gpio = pfile->private_data;

    if (_IOC_TYPE(cmd) != TOPBAND_GPIO_IOC_MAGIC) 
        return -EINVAL;
    if (_IOC_NR(cmd) > TOPBAND_GPIO_IOC_MAXNR) 
        return -EINVAL;

    if (_IOC_DIR(cmd) & _IOC_READ)
        ret = !access_ok(VERIFY_WRITE, (void *)arg, _IOC_SIZE(cmd));
    else if (_IOC_DIR(cmd) & _IOC_WRITE)
        ret = !access_ok(VERIFY_READ, (void *)arg, _IOC_SIZE(cmd));
    if (ret) 
        return -EFAULT;

    dev_info(&topband_gpio->platform_dev->dev,
                 "%s, (%x, %lx):\n", __func__, cmd,
                 arg);
    
    switch (cmd) {
        case TOPBAND_GPIO_IOC_SET_VALUE:
            if (copy_from_user(&data, (int *)arg, sizeof(int))) {
                dev_err(&topband_gpio->platform_dev->dev, 
                    "%s, copy from user failed\n", __func__);
                return -EFAULT;
            }
            ret = topband_gpio_ctrl(topband_gpio, data);
            break;
            
        default:
            return -EINVAL;
    }

    return ret;
}

static const struct file_operations topband_gpio_dev_fops = {
    .owner = THIS_MODULE,
    .open = topband_gpio_dev_open,
    .unlocked_ioctl = topband_gpio_dev_ioctl
};

static int topband_gpio_probe(struct platform_device *pdev)
{
    int ret = 0;
    struct topband_gpio_data *topband_gpio;

    topband_gpio = devm_kzalloc(&pdev->dev, sizeof(*topband_gpio), GFP_KERNEL);

    if(topband_gpio == NULL) {
        dev_err(&pdev->dev, "Failed alloc ts memory");
        return -ENOMEM;
    }

    if(pdev->dev.of_node) {
        ret = topband_gpio_parse_dt(&pdev->dev, topband_gpio);

        if(ret) {
            dev_err(&pdev->dev, "Failed parse dts\n");
            goto exit_free_data;
        }
    }

    topband_gpio->platform_dev = pdev;

    ret = topband_gpio_request_io_port(topband_gpio);

    if(ret < 0) {
        dev_err(&pdev->dev, "Failed request IO port\n");
        goto exit_free_data;
    }

    platform_set_drvdata(pdev, topband_gpio);

    topband_gpio->topband_gpio_device.minor = MISC_DYNAMIC_MINOR;
    topband_gpio->topband_gpio_device.name = "topband_gpio";
    topband_gpio->topband_gpio_device.fops = &topband_gpio_dev_fops;

    ret = misc_register(&topband_gpio->topband_gpio_device);
    if (ret) {
        dev_err(&pdev->dev, "Failed misc_register\n");
        goto exit_free_io_port;
    }

    dev_info(&pdev->dev, "%s, over\n", __func__);
    return 0;

exit_free_io_port:
    topband_gpio_free_io_port(topband_gpio);
    
exit_free_data:
    devm_kfree(&pdev->dev, topband_gpio);

    return ret;
}

static int topband_gpio_remove(struct platform_device *pdev)
{
    struct topband_gpio_data *topband_gpio = platform_get_drvdata(pdev);
    topband_gpio_free_io_port(topband_gpio);
    kfree(topband_gpio);

    return 0;
}

static const struct of_device_id topband_gpio_of_match[] = {
    { .compatible =  "topband,gpio"},
    {},
};

MODULE_DEVICE_TABLE(of, topband_gpio_of_match);

static struct platform_driver topband_gpio_driver = {
    .probe = topband_gpio_probe,
    .remove = topband_gpio_remove,
    .driver = {
        .name = TOPBAND_GPIO_NAME,
        .owner  = THIS_MODULE,
        .of_match_table = topband_gpio_of_match,
    },
};

module_platform_driver(topband_gpio_driver);

MODULE_AUTHOR("shenhb@topband.com");
MODULE_DESCRIPTION("Topband GPIO for application control");
MODULE_VERSION("1.0");
MODULE_LICENSE("GPL");

Kconfig

#
#  TOPBAND GPIO driver configuration
#
config TOPBAND_GPIO_DRIVER
       tristate  "Topband GPIO driver"    
       default n
       help
           Topband GPIO driver for application control

Makefile

obj-$(CONFIG_TOPBAND_GPIO_DRIVER) += topband_gpio.o
  1. 修改上级目录的Kconfig与Makefile
diff --git a/kernel/drivers/misc/Kconfig b/kernel/drivers/misc/Kconfig
index b72da72..f22ca49 100755
--- a/kernel/drivers/misc/Kconfig
+++ b/kernel/drivers/misc/Kconfig
@@ -577,4 +577,5 @@ source "drivers/misc/genwqe/Kconfig"
 source "drivers/misc/echo/Kconfig"
 source "drivers/misc/cxl/Kconfig"
 source "drivers/misc/n76e003/Kconfig"
+source "drivers/misc/topband_gpio/Kconfig"
diff --git a/kernel/drivers/misc/Makefile b/kernel/drivers/misc/Makefile
index e8d597b..2e97a69 100755
--- a/kernel/drivers/misc/Makefile
+++ b/kernel/drivers/misc/Makefile
@@ -62,3 +62,4 @@ obj-$(CONFIG_UID_SYS_STATS) += uid_sys_stats.o
 obj-$(CONFIG_MEMORY_STATE_TIME) += memory_state_time.o
 obj-$(CONFIG_USB_CAM_GPIO)     += usb_cam_gpio.o
 obj-$(CONFIG_N76E003_DRIVER)   += n76e003/
+obj-$(CONFIG_TOPBAND_GPIO_DRIVER)      += topband_gpio/
  1. 修改内核defconfig,将驱动编译至内核
diff --git a/kernel/arch/arm/configs/topband_pos_defconfig b/kernel/arch/arm/configs/topband_pos_defconfig
index 36c7947..cabe18c 100755
--- a/kernel/arch/arm/configs/topband_pos_defconfig
+++ b/kernel/arch/arm/configs/topband_pos_defconfig
@@ -715,4 +715,5 @@ CONFIG_CRYPTO_SHA256_ARM=y
 CONFIG_CRYPTO_SHA512_ARM=y
 CONFIG_CRYPTO_AES_ARM_BS=y
 
+CONFIG_TOPBAND_GPIO_DRIVER=y
  1. 配置dts文件
diff --git a/kernel/arch/arm/boot/dts/topband-pos-lvds-1280x800.dts b/kernel/arch/arm/boot/dts/topband-pos-lvds-12
index 45b0e45..0d99a97 100755
--- a/kernel/arch/arm/boot/dts/topband-pos-lvds-1280x800.dts
+++ b/kernel/arch/arm/boot/dts/topband-pos-lvds-1280x800.dts
@@ -96,6 +96,15 @@
                mwsensor,delay_time = <3>;
                mwsensor,is_poll = <1>;
        };
+       
+       topband_gpio: topband_gpio { 
+               status = "okay";
+               compatible = "topband,gpio";
+               gpio-1 = <&gpio7 8 GPIO_ACTIVE_HIGH>;
+               gpio-2 = <&gpio8 1 GPIO_ACTIVE_HIGH>;
+               gpio-3 = <&gpio8 2 GPIO_ACTIVE_HIGH>;
+               gpio-4 = <&gpio8 3 GPIO_ACTIVE_HIGH>;
+       };
 };
 
 &cpu0 {
  1. 修改设备文件权限为:0666,否则Hal层无权限访问
diff --git a/system/core/rootdir/ueventd.rc b/system/core/rootdir/ueventd.rc
old mode 100644
new mode 100755
index eadf219..f0a3609
--- a/system/core/rootdir/ueventd.rc
+++ b/system/core/rootdir/ueventd.rc
@@ -65,6 +65,9 @@ subsystem sound
 # kms driver for drm based gpu
 /dev/dri/*                0666   root       graphics
 
+# topband gpio driver
+/dev/topband_gpio         0666   system     system
+
 # these should not be world writable
 /dev/diag                 0660   radio      radio
 /dev/diag_arm9            0660   radio      radio
shenhb@dqrd01:~/code/rk3288$ 

Hal&Framework

修改文件:

    modified:   frameworks/base/Android.mk
    modified:   frameworks/base/core/java/android/app/SystemServiceRegistry.java
    modified:   frameworks/base/core/java/android/content/Context.java
    new file:   frameworks/base/core/java/android/os/IGpioService.aidl
    new file:   frameworks/base/core/java/android/os/SystemGpio.java
    new file:   frameworks/base/services/core/java/com/android/server/GpioService.java
    modified:   frameworks/base/services/core/jni/Android.mk
    new file:   frameworks/base/services/core/jni/com_android_server_GpioService.cpp
    modified:   frameworks/base/services/core/jni/onload.cpp
    modified:   frameworks/base/services/java/com/android/server/SystemServer.java
    new file:   hardware/libhardware/include/hardware/gpio_hal.h
    modified:   hardware/libhardware/modules/Android.mk
    new file:   hardware/libhardware/modules/gpio/Android.mk
    new file:   hardware/libhardware/modules/gpio/gpio_hal.c
    modified:   system/core/rootdir/ueventd.rc

HAL层

1.在hardware/libhardware/modules/下新建目录gpio目录,在gpio目录下创建Android.mk和gpio_hal.c

Android.mk

# Copyright (C) 2012 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#      http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, oftware
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or mplied.
# See the License for the specific language governing permissions nd
# limitations under the License.

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE := gpio.default

LOCAL_MODULE_RELATIVE_PATH := hw
LOCAL_PROPRIETARY_MODULE := true
LOCAL_SRC_FILES := gpio_hal.c
LOCAL_HEADER_LIBRARIES := libhardware_headers
LOCAL_SHARED_LIBRARIES := liblog libcutils libutils
LOCAL_MODULE_TAGS := optional

include $(BUILD_SHARED_LIBRARY)

gpio_hal.c

#include <hardware/hardware.h>
#include <cutils/log.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <hardware/gpio_hal.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <utils/Log.h>

#define DEVICE "/dev/topband_gpio"

#define GPIO_IOC_MAGIC  'f'
#define GPIO_IOC_SET_VALUE  _IOW(GPIO_IOC_MAGIC, 1, int)

static int fd;

static int gpio_close(struct hw_device_t* device)
{
    close(fd);
    return 0;
}

static int gpio_open(struct gpio_device_t* dev)
{
    fd = open(DEVICE, O_RDWR);
    ALOGI("gpio_open : %d", fd);

    if(fd >= 0) {
        return 0;
    } else {
        return -1;
    }
}

static int gpio_ctrl(struct gpio_device_t* dev, int gpio, int value)
{
    int ret = 0;
    int data = 0;
    
    if(fd < 0) {
        ret = gpio_open(dev);
        if (ret < 0) {
            return -1;
        }
    }

    data = ((gpio & 0x0f) << 4) | (value & 0x0f);
    ret = ioctl(fd, GPIO_IOC_SET_VALUE, &data);
    
    ALOGI("gpio_ctrl: gpio=%d, value=%d, data=%d, ret=%d", gpio, alue, data, ret);
    
    return ret;
}

static struct gpio_device_t gpio_dev = {
    .common = {
        .tag   = HARDWARE_DEVICE_TAG,
        .close = gpio_close,
    },
    .gpio_open  = gpio_open,
    .gpio_ctrl  = gpio_ctrl,
};

static int gpio_device_open(const struct hw_module_t* module, const har* id,
                           struct hw_device_t** device)
{
    *device = &gpio_dev;
    return 0;
}

static struct hw_module_methods_t gpio_module_methods = {
    .open = gpio_device_open,
};

struct hw_module_t HAL_MODULE_INFO_SYM = {
    .tag = HARDWARE_MODULE_TAG,
    .id = "gpio",
    .methods = &gpio_module_methods,
};

2.增加文件:hardware/libhardware/include/hardware/gpio_hal.h

#ifndef ANDROID_GPIO_INTERFACE_H
#define ANDROID_GPIO_INTERFACE_H

#include <stdint.h>
#include <sys/cdefs.h>
#include <sys/types.h>
#include <hardware/hardware.h>

__BEGIN_DECLS
struct gpio_device_t {
    struct hw_device_t common;

    int (*gpio_open)(struct gpio_device_t* dev);
    int (*gpio_ctrl)(struct gpio_device_t* dev, int gpio, int alue);
};

__END_DECLS^M

#endif  // ANDROID_GPIO_INTERFACE_H

3.修改父级Android.mk,将gpio模块加入编译

diff --git a/hardware/libhardware/modules/Android.mk b/hardware/libhardware/modules/Android.mk
old mode 100644
new mode 100755
index 462081d..9e33f41
--- a/hardware/libhardware/modules/Android.mk
+++ b/hardware/libhardware/modules/Android.mk
@@ -10,5 +10,6 @@ hardware_modules := \
     usbaudio \
     usbcamera \
     vehicle \
-    vr
+    vr \
+    gpio
 include $(call all-named-subdir-makefiles,$(hardware_modules))

JNI层

1.增加文件:frameworks/base/services/core/jni/com_android_server_GpioService.cpp

#include "jni.h"
#include "JNIHelp.h"
#include "android_runtime/AndroidRuntime.h"
#include <utils/misc.h>
#include <utils/Log.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <hardware/gpio_hal.h>

namespace android
{

static struct gpio_device_t* gpioDevice;

jint gpioOpen(JNIEnv *env, jobject cls)
{
    jint err;
    hw_module_t* module;
    hw_device_t* device;
    
    ALOGI("native gpioOpen ...");

    // hw_get_module finds the library by "gpio" (this is the id of al)
    err = hw_get_module("gpio", (hw_module_t const**)&module);
    if(err == 0) {
        // Get device : module->methods->open
        err = module->methods->open(module, NULL, &device);
        if(err == 0) {
            // Call gpio_open
            gpioDevice = (gpio_device_t *)device;
            return gpioDevice->gpio_open(gpioDevice);
        } else {
            return -1;
        }
    }

    return -1;
}

void gpioClose(JNIEnv *env, jobject cls)
{
    ALOGI("native gpioClose ...");
}

jint gpioCtrl(JNIEnv *env, jobject cls, jint gpio, jint value)
{
    ALOGI("native gpioCtrl gpio=%d, value=%d", gpio, value);
    return gpioDevice->gpio_ctrl(gpioDevice, gpio, value);
}

// Register native methods
static const JNINativeMethod methods[] = {
    {"native_gpioOpen", "()I", (void *)gpioOpen},
    {"native_gpioClose", "()V", (void *)gpioClose},
    {"native_gpioCtrl", "(II)I", (void *)gpioCtrl},
};
    
int register_android_server_GpioService(JNIEnv *env)
{
    // The Java method corresponding to the local method GpioService
    return jniRegisterNativeMethods(env, com/android/server/GpioService",
                                    methods, NELEM(methods));
}

}

2.修改:frameworks/base/services/core/jni/onload.cpp

diff --git a/frameworks/base/services/core/jni/onload.cpp b/frameworks/base/services/core/jni/onload.cpp
old mode 100644
new mode 100755
index 4d70384..08ddb64
--- a/frameworks/base/services/core/jni/onload.cpp
+++ b/frameworks/base/services/core/jni/onload.cpp
@@ -39,6 +39,7 @@ int register_android_server_UsbMidiDevice(JNIEnv* env);
 int register_android_server_UsbHostManager(JNIEnv* env);
 int register_android_server_vr_VrManagerService(JNIEnv* env);
 int register_android_server_VibratorService(JNIEnv* env);
+int register_android_server_GpioService(JNIEnv* env);
 int register_android_server_location_ContextHubService(JNIEnv* env);
 int register_android_server_location_GnssLocationProvider(JNIEnv* env);
 int register_android_server_connectivity_Vpn(JNIEnv* env);
@@ -82,6 +83,7 @@ extern "C" jint JNI_OnLoad(JavaVM* vm, void* /* reserved */)
     register_android_server_UsbHostManager(env);
     register_android_server_vr_VrManagerService(env);
     register_android_server_VibratorService(env);
+    register_android_server_GpioService(env);
     register_android_server_SystemServer(env);
     register_android_server_location_ContextHubService(env);
     register_android_server_location_GnssLocationProvider(env);

3.修改:frameworks/base/services/core/jni/Android.mk

diff --git a/frameworks/base/services/core/jni/Android.mk b/frameworks/base/services/core/jni/Android.mk
old mode 100644
new mode 100755
index b44362a..0d57546
--- a/frameworks/base/services/core/jni/Android.mk
+++ b/frameworks/base/services/core/jni/Android.mk
@@ -38,6 +38,7 @@ LOCAL_SRC_FILES += \
     $(LOCAL_REL_DIR)/com_android_server_UsbMidiDevice.cpp \
     $(LOCAL_REL_DIR)/com_android_server_UsbHostManager.cpp \
     $(LOCAL_REL_DIR)/com_android_server_VibratorService.cpp \
+    $(LOCAL_REL_DIR)/com_android_server_GpioService.cpp \
     $(LOCAL_REL_DIR)/com_android_server_PersistentDataBlockService.cpp \
     $(LOCAL_REL_DIR)/com_android_server_GraphicsStatsService.cpp \
     $(LOCAL_REL_DIR)/onload.cpp

Framework层(将GpioService注册到Framework)

1.增加文件:frameworks/base/core/java/android/os/IGpioService.aidl

package android.os;
 
/** {@hide} */
interface IGpioService
{
       int gpioCtrl(int gpio, int value);
}

2.增加文件:frameworks/base/services/core/java/com/android/server/GpioService.java

package com.android.server;
import android.os.IGpioService;

public class GpioService extends IGpioService.Stub
{
    private static final String TAG = "GpioService";

    /* call native c function to access hardware */
    public int gpioCtrl(int gpio, int value) throws ndroid.os.RemoteException
    {
        return native_gpioCtrl(gpio, value);
    }
    public GpioService()
    {
        native_gpioOpen();
    }

    public static native int native_gpioOpen();
    public static native void native_gpioClose();
    public static native int native_gpioCtrl(int gpio, int value);
}

3.修改:frameworks/base/Android.mk

diff --git a/frameworks/base/Android.mk b/frameworks/base/Android.mk
index 3018c4e..d301cfd 100755
--- a/frameworks/base/Android.mk
+++ b/frameworks/base/Android.mk
@@ -276,6 +276,7 @@ LOCAL_SRC_FILES += \
        core/java/android/os/IUpdateLock.aidl \
        core/java/android/os/IUserManager.aidl \
        core/java/android/os/IVibratorService.aidl \
+       core/java/android/os/IGpioService.aidl \
        core/java/android/os/storage/IStorageManager.aidl \
        core/java/android/os/storage/IStorageEventListener.aidl \
        core/java/android/os/storage/IStorageShutdownObserver.aidl \
  1. 修改:frameworks/base/services/java/com/android/server/SystemServer.java
diff --git a/frameworks/base/services/java/com/android/server/SystemServer.java b/frameworks/base/services/java/co
index fdf580e..8800516 100755
--- a/frameworks/base/services/java/com/android/server/SystemServer.java
+++ b/frameworks/base/services/java/com/android/server/SystemServer.java
@@ -795,6 +795,11 @@ public final class SystemServer {
             vibrator = new VibratorService(context);
             ServiceManager.addService("vibrator", vibrator);
             traceEnd();
+                      
+            traceBeginAndSlog("StartGpioService");
+            GpioService gpio = new GpioService();
+            ServiceManager.addService("gpio", gpio);
+            traceEnd();
 
             if (!disableConsumerIr) {
                 traceBeginAndSlog("StartConsumerIrService");

Framework(Application)

1.增加文件:frameworks/base/core/java/android/os/SystemGpio.java

package android.os;

import android.content.Context;
import android.media.AudioAttributes;
import android.util.Log;

/**
    Gpio implementation that controls the main system gpio.

    @hide
*/
public class SystemGpio
{
    private static final String TAG = "gpio";

    private final IGpioService mService;
    public SystemGpio()
    {
        mService = IGpioService.Stub.asInterface(
                       ServiceManager.getService("gpio"));
    }

    public SystemGpio(Context context)
    {
        mService = IGpioService.Stub.asInterface(
                       ServiceManager.getService("gpio"));
    }

    public void gpioCtrl(int gpio,int value)
    {
        try {
            mService.gpioCtrl(gpio, value);
        } catch(Exception e) {}
    }
}

2.修改:frameworks/base/core/java/android/content/Context.java

diff --git a/frameworks/base/core/java/android/content/Context.java b/frameworks/base/core/java/android/content/Co
index a31cb43..c18b848 100755
--- a/frameworks/base/core/java/android/content/Context.java
+++ b/frameworks/base/core/java/android/content/Context.java
@@ -3352,6 +3352,8 @@ public abstract class Context {
      * @see android.os.Vibrator
      */
     public static final String VIBRATOR_SERVICE = "vibrator";
+    
+    public static final String GPIO_SERVICE = "gpio";
 
     /**
      * Use with {@link #getSystemService} to retrieve a {@link

3.修改:frameworks/base/core/java/android/app/SystemServiceRegistry.java

diff --git a/frameworks/base/core/java/android/app/SystemServiceRegistry.java b/frameworks/base/core/java/android/
index e695936..ce1fe51 100755
--- a/frameworks/base/core/java/android/app/SystemServiceRegistry.java
+++ b/frameworks/base/core/java/android/app/SystemServiceRegistry.java
@@ -119,6 +119,7 @@ import android.os.SystemVibrator;
 import android.os.UserHandle;
 import android.os.UserManager;
 import android.os.Vibrator;
+import android.os.SystemGpio;
 import android.os.health.SystemHealthManager;
 import android.os.storage.StorageManager;
 import android.print.IPrintManager;
@@ -535,6 +536,14 @@ final class SystemServiceRegistry {
             public Vibrator createService(ContextImpl ctx) {
                 return new SystemVibrator(ctx);
             }});
+            
+            
+        registerService(Context.GPIO_SERVICE, SystemGpio.class,
+                new CachedServiceFetcher<SystemGpio>() {
+            @Override
+            public SystemGpio createService(ContextImpl ctx) {
+                return new SystemGpio(ctx);
+            }});
 
         registerService(Context.WALLPAPER_SERVICE, WallpaperManager.class,
                 new CachedServiceFetcher<WallpaperManager>() {

编译更新API

make update-api

APP

  1. 引入Android定制后的Framework jar包(out/target/common/obj/JAVA_LIBRARIES/framework_intermediates/classes.jar)
    1. 将classes.jar拷贝到app/libs/目录下
    2. 修改项目根目录build.gradle,在allprojects段内增加下面代码:
    allprojects {
        repositories {
            google()
            jcenter()
        }
    
        gradle.projectsEvaluated {
            tasks.withType(JavaCompile) {
                options.compilerArgs.add('-Xbootclasspath/p:app\\libs\\classes.jar')
            }
        }
    }
    
    1. 修改app/build.gradle,如下:
    android{
        defaultConfig {
            multiDexEnabled true
        }
    }
    
    dependencies {
        compileOnly files('libs/classes.jar')
    }
    
    preBuild {
        doLast {
            def imlFile = file(project.name + ".iml")
            println('Change ' + project.name + '.iml order')
            try {
                def parsedXml = (new XmlParser()).parse(imlFile)
                def jdkNode = parsedXml.component[1].orderEntry.find { it.'@type' == 'jdk' }
                parsedXml.component[1].remove(jdkNode)
                def sdkString = "Android API " + android.compileSdkVersion.substring("android-".length()) + " Platform"
                new groovy.util.Node(parsedXml.component[1], 'orderEntry', ['type': 'jdk', 'jdkName': sdkString, 'jdkType': 'Android SDK'])
                groovy.xml.XmlUtil.serialize(parsedXml, new FileOutputStream(imlFile))
            } catch (FileNotFoundException e) {
                // nop, iml not found
            }
        }
    }
    
  2. 系统GPIO Service类为SystemGpio,调用方式如下:
package com.ayst.item;

import android.annotation.SuppressLint;
import android.content.Context;
import android.os.SystemGpio;

public class GpioTest {
    private SystemGpio mGpioService;

    @SuppressLint("WrongConstant")
    public GpioTest(Context context) {
        mGpioService = (SystemGpio) context.getSystemService("gpio");
    }

    /**
     * GPIO control
     * @param gpio 1,2,3,4
     * @param value 0: Low 1: High
     */
    public void gpioCtrl(int gpio, int value) {
        if (null != mGpioService) {
            mGpioService.gpioCtrl(gpio, value);
        }
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容